2008-06-27 02:38:35 +08:00
|
|
|
//===--- SemaDeclAttr.cpp - Declaration Attribute Handling ----------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements decl-related attribute processing.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-08-26 06:03:47 +08:00
|
|
|
#include "clang/Sema/SemaInternal.h"
|
2010-01-10 20:58:08 +08:00
|
|
|
#include "TargetAttributesSema.h"
|
2008-06-27 02:38:35 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
2010-08-25 15:42:41 +08:00
|
|
|
#include "clang/AST/DeclCXX.h"
|
2008-08-11 14:23:49 +08:00
|
|
|
#include "clang/AST/DeclObjC.h"
|
|
|
|
#include "clang/AST/Expr.h"
|
2008-06-28 06:18:37 +08:00
|
|
|
#include "clang/Basic/TargetInfo.h"
|
2010-08-21 02:27:03 +08:00
|
|
|
#include "clang/Sema/DeclSpec.h"
|
2010-08-26 10:13:20 +08:00
|
|
|
#include "clang/Sema/DelayedDiagnostic.h"
|
2009-08-11 03:03:04 +08:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2008-06-27 02:38:35 +08:00
|
|
|
using namespace clang;
|
2010-08-26 10:13:20 +08:00
|
|
|
using namespace sema;
|
2008-06-27 02:38:35 +08:00
|
|
|
|
2008-06-29 08:16:31 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Helper functions
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-08-15 04:49:40 +08:00
|
|
|
static const FunctionType *getFunctionType(const Decl *d,
|
|
|
|
bool blocksToo = true) {
|
2008-06-27 02:38:35 +08:00
|
|
|
QualType Ty;
|
2009-08-15 04:49:40 +08:00
|
|
|
if (const ValueDecl *decl = dyn_cast<ValueDecl>(d))
|
2008-06-27 02:38:35 +08:00
|
|
|
Ty = decl->getType();
|
2009-08-15 04:49:40 +08:00
|
|
|
else if (const FieldDecl *decl = dyn_cast<FieldDecl>(d))
|
2008-06-27 02:38:35 +08:00
|
|
|
Ty = decl->getType();
|
2009-08-15 04:49:40 +08:00
|
|
|
else if (const TypedefDecl* decl = dyn_cast<TypedefDecl>(d))
|
2008-06-27 02:38:35 +08:00
|
|
|
Ty = decl->getUnderlyingType();
|
|
|
|
else
|
|
|
|
return 0;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
if (Ty->isFunctionPointerType())
|
2009-07-30 05:53:49 +08:00
|
|
|
Ty = Ty->getAs<PointerType>()->getPointeeType();
|
2009-05-19 01:39:25 +08:00
|
|
|
else if (blocksToo && Ty->isBlockPointerType())
|
2009-07-30 05:53:49 +08:00
|
|
|
Ty = Ty->getAs<BlockPointerType>()->getPointeeType();
|
2008-10-19 10:04:16 +08:00
|
|
|
|
2009-09-22 07:43:11 +08:00
|
|
|
return Ty->getAs<FunctionType>();
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2008-09-26 12:12:28 +08:00
|
|
|
// FIXME: We should provide an abstraction around a method or function
|
|
|
|
// to provide the following bits of information.
|
|
|
|
|
2009-12-21 07:11:08 +08:00
|
|
|
/// isFunction - Return true if the given decl has function
|
2009-08-15 04:49:40 +08:00
|
|
|
/// type (function or function-typed variable).
|
|
|
|
static bool isFunction(const Decl *d) {
|
|
|
|
return getFunctionType(d, false) != NULL;
|
|
|
|
}
|
|
|
|
|
2008-10-19 10:04:16 +08:00
|
|
|
/// isFunctionOrMethod - Return true if the given decl has function
|
|
|
|
/// type (function or function-typed variable) or an Objective-C
|
|
|
|
/// method.
|
2009-08-15 04:49:40 +08:00
|
|
|
static bool isFunctionOrMethod(const Decl *d) {
|
|
|
|
return isFunction(d)|| isa<ObjCMethodDecl>(d);
|
2008-10-19 10:04:16 +08:00
|
|
|
}
|
2008-09-26 12:12:28 +08:00
|
|
|
|
2009-05-16 07:15:03 +08:00
|
|
|
/// isFunctionOrMethodOrBlock - Return true if the given decl has function
|
|
|
|
/// type (function or function-typed variable) or an Objective-C
|
|
|
|
/// method or a block.
|
2009-08-15 04:49:40 +08:00
|
|
|
static bool isFunctionOrMethodOrBlock(const Decl *d) {
|
2009-05-16 07:15:03 +08:00
|
|
|
if (isFunctionOrMethod(d))
|
|
|
|
return true;
|
|
|
|
// check for block is more involved.
|
|
|
|
if (const VarDecl *V = dyn_cast<VarDecl>(d)) {
|
|
|
|
QualType Ty = V->getType();
|
|
|
|
return Ty->isBlockPointerType();
|
|
|
|
}
|
2009-05-20 01:08:59 +08:00
|
|
|
return isa<BlockDecl>(d);
|
2009-05-16 07:15:03 +08:00
|
|
|
}
|
|
|
|
|
2008-10-19 10:04:16 +08:00
|
|
|
/// hasFunctionProto - Return true if the given decl has a argument
|
|
|
|
/// information. This decl should have already passed
|
2009-05-16 07:15:03 +08:00
|
|
|
/// isFunctionOrMethod or isFunctionOrMethodOrBlock.
|
2009-08-15 04:49:40 +08:00
|
|
|
static bool hasFunctionProto(const Decl *d) {
|
2009-05-16 07:15:03 +08:00
|
|
|
if (const FunctionType *FnTy = getFunctionType(d))
|
2009-02-27 07:50:07 +08:00
|
|
|
return isa<FunctionProtoType>(FnTy);
|
2009-05-16 07:15:03 +08:00
|
|
|
else {
|
2009-05-20 01:08:59 +08:00
|
|
|
assert(isa<ObjCMethodDecl>(d) || isa<BlockDecl>(d));
|
2008-10-19 10:04:16 +08:00
|
|
|
return true;
|
|
|
|
}
|
2008-09-26 12:12:28 +08:00
|
|
|
}
|
|
|
|
|
2008-10-19 10:04:16 +08:00
|
|
|
/// getFunctionOrMethodNumArgs - Return number of function or method
|
|
|
|
/// arguments. It is an error to call this on a K&R function (use
|
|
|
|
/// hasFunctionProto first).
|
2009-08-15 04:49:40 +08:00
|
|
|
static unsigned getFunctionOrMethodNumArgs(const Decl *d) {
|
2009-02-21 02:43:26 +08:00
|
|
|
if (const FunctionType *FnTy = getFunctionType(d))
|
2009-02-27 07:50:07 +08:00
|
|
|
return cast<FunctionProtoType>(FnTy)->getNumArgs();
|
2009-05-20 01:08:59 +08:00
|
|
|
if (const BlockDecl *BD = dyn_cast<BlockDecl>(d))
|
|
|
|
return BD->getNumParams();
|
2009-02-21 02:43:26 +08:00
|
|
|
return cast<ObjCMethodDecl>(d)->param_size();
|
2008-09-26 12:12:28 +08:00
|
|
|
}
|
|
|
|
|
2009-08-15 04:49:40 +08:00
|
|
|
static QualType getFunctionOrMethodArgType(const Decl *d, unsigned Idx) {
|
2009-02-21 02:43:26 +08:00
|
|
|
if (const FunctionType *FnTy = getFunctionType(d))
|
2009-02-27 07:50:07 +08:00
|
|
|
return cast<FunctionProtoType>(FnTy)->getArgType(Idx);
|
2009-05-20 01:08:59 +08:00
|
|
|
if (const BlockDecl *BD = dyn_cast<BlockDecl>(d))
|
|
|
|
return BD->getParamDecl(Idx)->getType();
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-02-21 02:43:26 +08:00
|
|
|
return cast<ObjCMethodDecl>(d)->param_begin()[Idx]->getType();
|
2008-09-26 12:12:28 +08:00
|
|
|
}
|
|
|
|
|
2009-08-15 04:49:40 +08:00
|
|
|
static QualType getFunctionOrMethodResultType(const Decl *d) {
|
2009-05-21 01:41:43 +08:00
|
|
|
if (const FunctionType *FnTy = getFunctionType(d))
|
|
|
|
return cast<FunctionProtoType>(FnTy)->getResultType();
|
|
|
|
return cast<ObjCMethodDecl>(d)->getResultType();
|
|
|
|
}
|
|
|
|
|
2009-08-15 04:49:40 +08:00
|
|
|
static bool isFunctionOrMethodVariadic(const Decl *d) {
|
2008-10-19 10:04:16 +08:00
|
|
|
if (const FunctionType *FnTy = getFunctionType(d)) {
|
2009-02-27 07:50:07 +08:00
|
|
|
const FunctionProtoType *proto = cast<FunctionProtoType>(FnTy);
|
2008-09-26 12:12:28 +08:00
|
|
|
return proto->isVariadic();
|
2009-05-20 01:08:59 +08:00
|
|
|
} else if (const BlockDecl *BD = dyn_cast<BlockDecl>(d))
|
2010-04-30 00:48:58 +08:00
|
|
|
return BD->isVariadic();
|
2009-05-20 01:08:59 +08:00
|
|
|
else {
|
2008-09-26 12:12:28 +08:00
|
|
|
return cast<ObjCMethodDecl>(d)->isVariadic();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
static inline bool isNSStringType(QualType T, ASTContext &Ctx) {
|
2009-09-22 07:43:11 +08:00
|
|
|
const ObjCObjectPointerType *PT = T->getAs<ObjCObjectPointerType>();
|
2008-07-27 06:17:49 +08:00
|
|
|
if (!PT)
|
2008-06-27 02:38:35 +08:00
|
|
|
return false;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-05-18 05:00:27 +08:00
|
|
|
ObjCInterfaceDecl *Cls = PT->getObjectType()->getInterface();
|
|
|
|
if (!Cls)
|
2008-06-27 02:38:35 +08:00
|
|
|
return false;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-05-18 05:00:27 +08:00
|
|
|
IdentifierInfo* ClsName = Cls->getIdentifier();
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
// FIXME: Should we walk the chain of classes?
|
|
|
|
return ClsName == &Ctx.Idents.get("NSString") ||
|
|
|
|
ClsName == &Ctx.Idents.get("NSMutableString");
|
|
|
|
}
|
|
|
|
|
2008-09-26 11:32:58 +08:00
|
|
|
static inline bool isCFStringType(QualType T, ASTContext &Ctx) {
|
2009-07-30 05:53:49 +08:00
|
|
|
const PointerType *PT = T->getAs<PointerType>();
|
2008-09-26 11:32:58 +08:00
|
|
|
if (!PT)
|
|
|
|
return false;
|
|
|
|
|
2009-07-30 05:53:49 +08:00
|
|
|
const RecordType *RT = PT->getPointeeType()->getAs<RecordType>();
|
2008-09-26 11:32:58 +08:00
|
|
|
if (!RT)
|
|
|
|
return false;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-09-26 11:32:58 +08:00
|
|
|
const RecordDecl *RD = RT->getDecl();
|
2010-05-12 05:36:43 +08:00
|
|
|
if (RD->getTagKind() != TTK_Struct)
|
2008-09-26 11:32:58 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return RD->getIdentifier() == &Ctx.Idents.get("__CFString");
|
|
|
|
}
|
|
|
|
|
2008-06-29 08:16:31 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Attribute Implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2008-08-01 06:40:48 +08:00
|
|
|
// FIXME: All this manual attribute parsing code is gross. At the
|
|
|
|
// least add some helper functions to check most argument patterns (#
|
|
|
|
// and types of args).
|
|
|
|
|
2009-07-25 03:02:52 +08:00
|
|
|
static void HandleExtVectorTypeAttr(Scope *scope, Decl *d,
|
2009-06-18 05:51:59 +08:00
|
|
|
const AttributeList &Attr, Sema &S) {
|
2008-06-29 07:36:30 +08:00
|
|
|
TypedefDecl *tDecl = dyn_cast<TypedefDecl>(d);
|
|
|
|
if (tDecl == 0) {
|
2008-06-29 08:43:07 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_typecheck_ext_vector_not_typedef);
|
2008-06-29 07:36:30 +08:00
|
|
|
return;
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
QualType curType = tDecl->getUnderlyingType();
|
2009-06-18 05:51:59 +08:00
|
|
|
|
|
|
|
Expr *sizeExpr;
|
|
|
|
|
|
|
|
// Special case where the argument is a template id.
|
|
|
|
if (Attr.getParameterName()) {
|
2009-11-25 03:00:30 +08:00
|
|
|
CXXScopeSpec SS;
|
|
|
|
UnqualifiedId id;
|
|
|
|
id.setIdentifier(Attr.getParameterName(), Attr.getLoc());
|
|
|
|
sizeExpr = S.ActOnIdExpression(scope, SS, id, false, false).takeAs<Expr>();
|
2009-06-18 05:51:59 +08:00
|
|
|
} else {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 1) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
sizeExpr = static_cast<Expr *>(Attr.getArg(0));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
2009-06-18 05:51:59 +08:00
|
|
|
|
|
|
|
// Instantiate/Install the vector type, and let Sema build the type for us.
|
|
|
|
// This will run the reguired checks.
|
2010-08-24 07:25:46 +08:00
|
|
|
QualType T = S.BuildExtVectorType(curType, sizeExpr, Attr.getLoc());
|
2009-06-18 05:51:59 +08:00
|
|
|
if (!T.isNull()) {
|
2009-10-24 16:00:42 +08:00
|
|
|
// FIXME: preserve the old source info.
|
2009-12-07 10:54:59 +08:00
|
|
|
tDecl->setTypeSourceInfo(S.Context.getTrivialTypeSourceInfo(T));
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-06-18 05:51:59 +08:00
|
|
|
// Remember this typedef decl, we will need it later for diagnostics.
|
|
|
|
S.ExtVectorDecls.push_back(tDecl);
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-29 08:43:07 +08:00
|
|
|
static void HandlePackedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() > 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
if (TagDecl *TD = dyn_cast<TagDecl>(d))
|
2010-08-19 07:23:40 +08:00
|
|
|
TD->addAttr(::new (S.Context) PackedAttr(Attr.getLoc(), S.Context));
|
2008-06-27 02:38:35 +08:00
|
|
|
else if (FieldDecl *FD = dyn_cast<FieldDecl>(d)) {
|
|
|
|
// If the alignment is less than or equal to 8 bits, the packed attribute
|
|
|
|
// has no effect.
|
|
|
|
if (!FD->getType()->isIncompleteType() &&
|
2008-06-29 08:43:07 +08:00
|
|
|
S.Context.getTypeAlign(FD->getType()) <= 8)
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_ignored_for_field_of_type)
|
2008-11-24 05:45:46 +08:00
|
|
|
<< Attr.getName() << FD->getType();
|
2008-06-27 02:38:35 +08:00
|
|
|
else
|
2010-08-19 07:23:40 +08:00
|
|
|
FD->addAttr(::new (S.Context) PackedAttr(Attr.getLoc(), S.Context));
|
2008-06-27 02:38:35 +08:00
|
|
|
} else
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << Attr.getName();
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2010-02-18 11:08:58 +08:00
|
|
|
static void HandleIBAction(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-07-16 06:26:48 +08:00
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() > 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-07-16 06:26:48 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-02-18 11:08:58 +08:00
|
|
|
// The IBAction attributes only apply to instance methods.
|
|
|
|
if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(d))
|
|
|
|
if (MD->isInstanceMethod()) {
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) IBActionAttr(Attr.getLoc(), S.Context));
|
2010-02-18 11:08:58 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_ibaction) << Attr.getName();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleIBOutlet(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() > 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The IBOutlet attributes only apply to instance variables of
|
2010-02-17 10:37:45 +08:00
|
|
|
// Objective-C classes.
|
|
|
|
if (isa<ObjCIvarDecl>(d) || isa<ObjCPropertyDecl>(d)) {
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) IBOutletAttr(Attr.getLoc(), S.Context));
|
2010-02-18 11:08:58 +08:00
|
|
|
return;
|
2010-02-17 10:37:45 +08:00
|
|
|
}
|
2010-02-18 11:08:58 +08:00
|
|
|
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_iboutlet) << Attr.getName();
|
2008-07-16 06:26:48 +08:00
|
|
|
}
|
|
|
|
|
2010-05-20 01:38:06 +08:00
|
|
|
static void HandleIBOutletCollection(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
|
|
|
|
// The iboutletcollection attribute can have zero or one arguments.
|
2010-08-18 04:23:12 +08:00
|
|
|
if (Attr.getParameterName() && Attr.getNumArgs() > 0) {
|
2010-05-20 01:38:06 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The IBOutletCollection attributes only apply to instance variables of
|
|
|
|
// Objective-C classes.
|
|
|
|
if (!(isa<ObjCIvarDecl>(d) || isa<ObjCPropertyDecl>(d))) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_iboutlet) << Attr.getName();
|
|
|
|
return;
|
|
|
|
}
|
2010-08-18 05:39:27 +08:00
|
|
|
if (const ValueDecl *VD = dyn_cast<ValueDecl>(d))
|
|
|
|
if (!VD->getType()->getAs<ObjCObjectPointerType>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_iboutletcollection_object_type)
|
|
|
|
<< VD->getType() << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (const ObjCPropertyDecl *PD = dyn_cast<ObjCPropertyDecl>(d))
|
|
|
|
if (!PD->getType()->getAs<ObjCObjectPointerType>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_iboutletcollection_object_type)
|
|
|
|
<< PD->getType() << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-18 04:23:12 +08:00
|
|
|
IdentifierInfo *II = Attr.getParameterName();
|
|
|
|
if (!II)
|
|
|
|
II = &S.Context.Idents.get("id");
|
2010-08-18 05:39:27 +08:00
|
|
|
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType TypeRep = S.getTypeName(*II, Attr.getLoc(),
|
2010-08-18 04:23:12 +08:00
|
|
|
S.getScopeForContext(d->getDeclContext()->getParent()));
|
|
|
|
if (!TypeRep) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_iboutletcollection_type) << II;
|
|
|
|
return;
|
|
|
|
}
|
2010-08-24 13:47:05 +08:00
|
|
|
QualType QT = TypeRep.get();
|
2010-08-18 04:23:12 +08:00
|
|
|
// Diagnose use of non-object type in iboutletcollection attribute.
|
|
|
|
// FIXME. Gnu attribute extension ignores use of builtin types in
|
|
|
|
// attributes. So, __attribute__((iboutletcollection(char))) will be
|
|
|
|
// treated as __attribute__((iboutletcollection())).
|
|
|
|
if (!QT->isObjCIdType() && !QT->isObjCClassType() &&
|
|
|
|
!QT->isObjCObjectType()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_iboutletcollection_type) << II;
|
|
|
|
return;
|
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) IBOutletCollectionAttr(Attr.getLoc(), S.Context,
|
|
|
|
QT));
|
2010-05-20 01:38:06 +08:00
|
|
|
}
|
|
|
|
|
2008-07-22 05:53:04 +08:00
|
|
|
static void HandleNonNullAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2009-07-25 03:02:52 +08:00
|
|
|
// GCC ignores the nonnull attribute on K&R style function prototypes, so we
|
|
|
|
// ignore it as well
|
2008-10-19 10:04:16 +08:00
|
|
|
if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2008-07-22 05:53:04 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-10-19 10:04:16 +08:00
|
|
|
unsigned NumArgs = getFunctionOrMethodNumArgs(d);
|
2008-07-22 05:53:04 +08:00
|
|
|
|
|
|
|
// The nonnull attribute only applies to pointers.
|
|
|
|
llvm::SmallVector<unsigned, 10> NonNullArgs;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-07-22 05:53:04 +08:00
|
|
|
for (AttributeList::arg_iterator I=Attr.arg_begin(),
|
|
|
|
E=Attr.arg_end(); I!=E; ++I) {
|
2009-07-25 03:02:52 +08:00
|
|
|
|
|
|
|
|
2008-07-22 05:53:04 +08:00
|
|
|
// The argument must be an integer constant expression.
|
2008-12-05 03:38:33 +08:00
|
|
|
Expr *Ex = static_cast<Expr *>(*I);
|
2008-07-22 05:53:04 +08:00
|
|
|
llvm::APSInt ArgNum(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (Ex->isTypeDependent() || Ex->isValueDependent() ||
|
|
|
|
!Ex->isIntegerConstantExpr(ArgNum, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
|
|
|
|
<< "nonnull" << Ex->getSourceRange();
|
2008-07-22 05:53:04 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-07-22 05:53:04 +08:00
|
|
|
unsigned x = (unsigned) ArgNum.getZExtValue();
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-07-22 05:53:04 +08:00
|
|
|
if (x < 1 || x > NumArgs) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
|
2008-11-19 15:22:31 +08:00
|
|
|
<< "nonnull" << I.getArgNum() << Ex->getSourceRange();
|
2008-07-22 05:53:04 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-07-22 06:09:15 +08:00
|
|
|
--x;
|
2008-07-22 05:53:04 +08:00
|
|
|
|
|
|
|
// Is the function argument a pointer type?
|
2009-07-25 03:02:52 +08:00
|
|
|
QualType T = getFunctionOrMethodArgType(d, x);
|
2009-07-16 07:23:54 +08:00
|
|
|
if (!T->isAnyPointerType() && !T->isBlockPointerType()) {
|
2008-07-22 05:53:04 +08:00
|
|
|
// FIXME: Should also highlight argument in decl.
|
2010-08-13 02:48:43 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_nonnull_pointers_only)
|
2008-11-19 13:08:23 +08:00
|
|
|
<< "nonnull" << Ex->getSourceRange();
|
2008-09-02 03:57:52 +08:00
|
|
|
continue;
|
2008-07-22 05:53:04 +08:00
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-07-22 05:53:04 +08:00
|
|
|
NonNullArgs.push_back(x);
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
|
|
|
// If no arguments were specified to __attribute__((nonnull)) then all pointer
|
|
|
|
// arguments have a nonnull attribute.
|
2008-09-02 03:57:52 +08:00
|
|
|
if (NonNullArgs.empty()) {
|
2008-11-18 14:52:58 +08:00
|
|
|
for (unsigned I = 0, E = getFunctionOrMethodNumArgs(d); I != E; ++I) {
|
|
|
|
QualType T = getFunctionOrMethodArgType(d, I);
|
2009-07-16 07:23:54 +08:00
|
|
|
if (T->isAnyPointerType() || T->isBlockPointerType())
|
2008-10-19 10:04:16 +08:00
|
|
|
NonNullArgs.push_back(I);
|
2010-09-28 06:42:37 +08:00
|
|
|
else if (const RecordType *UT = T->getAsUnionType()) {
|
|
|
|
if (UT && UT->getDecl()->hasAttr<TransparentUnionAttr>()) {
|
|
|
|
RecordDecl *UD = UT->getDecl();
|
|
|
|
for (RecordDecl::field_iterator it = UD->field_begin(),
|
|
|
|
itend = UD->field_end(); it != itend; ++it) {
|
|
|
|
T = it->getType();
|
|
|
|
if (T->isAnyPointerType() || T->isBlockPointerType()) {
|
|
|
|
NonNullArgs.push_back(I);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-11-18 14:52:58 +08:00
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-09-09 09:17:32 +08:00
|
|
|
// No pointer arguments? The attribute in this case is
|
|
|
|
// trivially satisfied.
|
2010-09-28 03:05:51 +08:00
|
|
|
if (NonNullArgs.empty()) {
|
|
|
|
// Warn the trivial case only if attribute is not coming from a
|
|
|
|
// macro instantiation.
|
|
|
|
if (Attr.getLoc().isFileID())
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_nonnull_no_pointers);
|
2008-09-02 03:57:52 +08:00
|
|
|
return;
|
2010-09-28 03:05:51 +08:00
|
|
|
}
|
2008-07-22 05:53:04 +08:00
|
|
|
}
|
2008-09-02 03:57:52 +08:00
|
|
|
|
|
|
|
unsigned* start = &NonNullArgs[0];
|
|
|
|
unsigned size = NonNullArgs.size();
|
2010-07-31 09:52:11 +08:00
|
|
|
llvm::array_pod_sort(start, start + size);
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) NonNullAttr(Attr.getLoc(), S.Context, start,
|
|
|
|
size));
|
2008-07-22 05:53:04 +08:00
|
|
|
}
|
|
|
|
|
2010-07-31 09:52:11 +08:00
|
|
|
static void HandleOwnershipAttr(Decl *d, const AttributeList &AL, Sema &S) {
|
|
|
|
// This attribute must be applied to a function declaration.
|
|
|
|
// The first argument to the attribute must be a string,
|
|
|
|
// the name of the resource, for example "malloc".
|
|
|
|
// The following arguments must be argument indexes, the arguments must be
|
|
|
|
// of integer type for Returns, otherwise of pointer type.
|
|
|
|
// The difference between Holds and Takes is that a pointer may still be used
|
2010-08-12 16:54:03 +08:00
|
|
|
// after being held. free() should be __attribute((ownership_takes)), whereas
|
|
|
|
// a list append function may well be __attribute((ownership_holds)).
|
2010-07-31 09:52:11 +08:00
|
|
|
|
|
|
|
if (!AL.getParameterName()) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attribute_argument_n_not_string)
|
|
|
|
<< AL.getName()->getName() << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Figure out our Kind, and check arguments while we're at it.
|
2010-08-19 07:23:40 +08:00
|
|
|
OwnershipAttr::OwnershipKind K;
|
2010-08-12 16:54:03 +08:00
|
|
|
switch (AL.getKind()) {
|
|
|
|
case AttributeList::AT_ownership_takes:
|
2010-08-19 07:23:40 +08:00
|
|
|
K = OwnershipAttr::Takes;
|
2010-07-31 09:52:11 +08:00
|
|
|
if (AL.getNumArgs() < 1) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attribute_wrong_number_arguments) << 2;
|
|
|
|
return;
|
|
|
|
}
|
2010-08-12 16:54:03 +08:00
|
|
|
break;
|
|
|
|
case AttributeList::AT_ownership_holds:
|
2010-08-19 07:23:40 +08:00
|
|
|
K = OwnershipAttr::Holds;
|
2010-07-31 09:52:11 +08:00
|
|
|
if (AL.getNumArgs() < 1) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attribute_wrong_number_arguments) << 2;
|
|
|
|
return;
|
|
|
|
}
|
2010-08-12 16:54:03 +08:00
|
|
|
break;
|
|
|
|
case AttributeList::AT_ownership_returns:
|
2010-08-19 07:23:40 +08:00
|
|
|
K = OwnershipAttr::Returns;
|
2010-07-31 09:52:11 +08:00
|
|
|
if (AL.getNumArgs() > 1) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attribute_wrong_number_arguments)
|
|
|
|
<< AL.getNumArgs() + 1;
|
|
|
|
return;
|
|
|
|
}
|
2010-08-12 16:54:03 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// This should never happen given how we are called.
|
|
|
|
llvm_unreachable("Unknown ownership attribute");
|
2010-07-31 09:52:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!isFunction(d) || !hasFunctionProto(d)) {
|
|
|
|
S.Diag(AL.getLoc(), diag::warn_attribute_wrong_decl_type) << AL.getName()
|
|
|
|
<< 0 /*function*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned NumArgs = getFunctionOrMethodNumArgs(d);
|
|
|
|
|
|
|
|
llvm::StringRef Module = AL.getParameterName()->getName();
|
|
|
|
|
|
|
|
// Normalize the argument, __foo__ becomes foo.
|
|
|
|
if (Module.startswith("__") && Module.endswith("__"))
|
|
|
|
Module = Module.substr(2, Module.size() - 4);
|
|
|
|
|
|
|
|
llvm::SmallVector<unsigned, 10> OwnershipArgs;
|
|
|
|
|
2010-08-12 16:54:03 +08:00
|
|
|
for (AttributeList::arg_iterator I = AL.arg_begin(), E = AL.arg_end(); I != E;
|
|
|
|
++I) {
|
2010-07-31 09:52:11 +08:00
|
|
|
|
|
|
|
Expr *IdxExpr = static_cast<Expr *>(*I);
|
|
|
|
llvm::APSInt ArgNum(32);
|
|
|
|
if (IdxExpr->isTypeDependent() || IdxExpr->isValueDependent()
|
|
|
|
|| !IdxExpr->isIntegerConstantExpr(ArgNum, S.Context)) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attribute_argument_not_int)
|
|
|
|
<< AL.getName()->getName() << IdxExpr->getSourceRange();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned x = (unsigned) ArgNum.getZExtValue();
|
|
|
|
|
|
|
|
if (x > NumArgs || x < 1) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attribute_argument_out_of_bounds)
|
|
|
|
<< AL.getName()->getName() << x << IdxExpr->getSourceRange();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
--x;
|
|
|
|
switch (K) {
|
2010-08-19 07:23:40 +08:00
|
|
|
case OwnershipAttr::Takes:
|
|
|
|
case OwnershipAttr::Holds: {
|
2010-07-31 09:52:11 +08:00
|
|
|
// Is the function argument a pointer type?
|
|
|
|
QualType T = getFunctionOrMethodArgType(d, x);
|
|
|
|
if (!T->isAnyPointerType() && !T->isBlockPointerType()) {
|
|
|
|
// FIXME: Should also highlight argument in decl.
|
|
|
|
S.Diag(AL.getLoc(), diag::err_ownership_type)
|
2010-08-19 07:23:40 +08:00
|
|
|
<< ((K==OwnershipAttr::Takes)?"ownership_takes":"ownership_holds")
|
2010-07-31 09:52:11 +08:00
|
|
|
<< "pointer"
|
|
|
|
<< IdxExpr->getSourceRange();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
case OwnershipAttr::Returns: {
|
2010-07-31 09:52:11 +08:00
|
|
|
if (AL.getNumArgs() > 1) {
|
|
|
|
// Is the function argument an integer type?
|
|
|
|
Expr *IdxExpr = static_cast<Expr *>(AL.getArg(0));
|
|
|
|
llvm::APSInt ArgNum(32);
|
|
|
|
if (IdxExpr->isTypeDependent() || IdxExpr->isValueDependent()
|
|
|
|
|| !IdxExpr->isIntegerConstantExpr(ArgNum, S.Context)) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_ownership_type)
|
|
|
|
<< "ownership_returns" << "integer"
|
|
|
|
<< IdxExpr->getSourceRange();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2010-08-12 16:54:03 +08:00
|
|
|
default:
|
|
|
|
llvm_unreachable("Unknown ownership attribute");
|
2010-07-31 09:52:11 +08:00
|
|
|
} // switch
|
|
|
|
|
|
|
|
// Check we don't have a conflict with another ownership attribute.
|
2010-08-19 07:23:40 +08:00
|
|
|
for (specific_attr_iterator<OwnershipAttr>
|
|
|
|
i = d->specific_attr_begin<OwnershipAttr>(),
|
|
|
|
e = d->specific_attr_end<OwnershipAttr>();
|
|
|
|
i != e; ++i) {
|
|
|
|
if ((*i)->getOwnKind() != K) {
|
|
|
|
for (const unsigned *I = (*i)->args_begin(), *E = (*i)->args_end();
|
|
|
|
I!=E; ++I) {
|
|
|
|
if (x == *I) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attributes_are_not_compatible)
|
|
|
|
<< AL.getName()->getName() << "ownership_*";
|
2010-07-31 09:52:11 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
OwnershipArgs.push_back(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned* start = OwnershipArgs.data();
|
|
|
|
unsigned size = OwnershipArgs.size();
|
|
|
|
llvm::array_pod_sort(start, start + size);
|
2010-08-19 07:23:40 +08:00
|
|
|
|
|
|
|
if (K != OwnershipAttr::Returns && OwnershipArgs.empty()) {
|
|
|
|
S.Diag(AL.getLoc(), diag::err_attribute_wrong_number_arguments) << 2;
|
|
|
|
return;
|
2010-07-31 09:52:11 +08:00
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
|
|
|
|
d->addAttr(::new (S.Context) OwnershipAttr(AL.getLoc(), S.Context, K, Module,
|
|
|
|
start, size));
|
2010-07-31 09:52:11 +08:00
|
|
|
}
|
|
|
|
|
2010-02-24 06:00:30 +08:00
|
|
|
static bool isStaticVarOrStaticFunciton(Decl *D) {
|
|
|
|
if (VarDecl *VD = dyn_cast<VarDecl>(D))
|
2010-08-26 11:08:43 +08:00
|
|
|
return VD->getStorageClass() == SC_Static;
|
2010-02-24 06:00:30 +08:00
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D))
|
2010-08-26 11:08:43 +08:00
|
|
|
return FD->getStorageClass() == SC_Static;
|
2010-02-24 06:00:30 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleWeakRefAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// Check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() > 1) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// gcc rejects
|
|
|
|
// class c {
|
|
|
|
// static int a __attribute__((weakref ("v2")));
|
|
|
|
// static int b() __attribute__((weakref ("f3")));
|
|
|
|
// };
|
|
|
|
// and ignores the attributes of
|
|
|
|
// void f(void) {
|
|
|
|
// static int a __attribute__((weakref ("v2")));
|
|
|
|
// }
|
|
|
|
// we reject them
|
2010-08-31 08:36:30 +08:00
|
|
|
const DeclContext *Ctx = d->getDeclContext()->getRedeclContext();
|
|
|
|
if (!Ctx->isFileContext()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_weakref_not_global_context) <<
|
|
|
|
dyn_cast<NamedDecl>(d)->getNameAsString();
|
|
|
|
return;
|
2010-02-24 06:00:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// The GCC manual says
|
|
|
|
//
|
|
|
|
// At present, a declaration to which `weakref' is attached can only
|
|
|
|
// be `static'.
|
|
|
|
//
|
|
|
|
// It also says
|
|
|
|
//
|
|
|
|
// Without a TARGET,
|
|
|
|
// given as an argument to `weakref' or to `alias', `weakref' is
|
|
|
|
// equivalent to `weak'.
|
|
|
|
//
|
|
|
|
// gcc 4.4.1 will accept
|
|
|
|
// int a7 __attribute__((weakref));
|
|
|
|
// as
|
|
|
|
// int a7 __attribute__((weak));
|
|
|
|
// This looks like a bug in gcc. We reject that for now. We should revisit
|
|
|
|
// it if this behaviour is actually used.
|
|
|
|
|
|
|
|
if (!isStaticVarOrStaticFunciton(d)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_weakref_not_static) <<
|
|
|
|
dyn_cast<NamedDecl>(d)->getNameAsString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// GCC rejects
|
|
|
|
// static ((alias ("y"), weakref)).
|
|
|
|
// Should we? How to check that weakref is before or after alias?
|
|
|
|
|
|
|
|
if (Attr.getNumArgs() == 1) {
|
|
|
|
Expr *Arg = static_cast<Expr*>(Attr.getArg(0));
|
|
|
|
Arg = Arg->IgnoreParenCasts();
|
|
|
|
StringLiteral *Str = dyn_cast<StringLiteral>(Arg);
|
|
|
|
|
|
|
|
if (Str == 0 || Str->isWide()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
|
|
|
|
<< "weakref" << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// GCC will accept anything as the argument of weakref. Should we
|
|
|
|
// check for an existing decl?
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) AliasAttr(Attr.getLoc(), S.Context, Str->getString()));
|
2010-02-24 06:00:30 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) WeakRefAttr(Attr.getLoc(), S.Context));
|
2010-02-24 06:00:30 +08:00
|
|
|
}
|
|
|
|
|
2008-06-29 08:43:07 +08:00
|
|
|
static void HandleAliasAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 1) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-29 07:36:30 +08:00
|
|
|
Expr *Arg = static_cast<Expr*>(Attr.getArg(0));
|
2008-06-27 02:38:35 +08:00
|
|
|
Arg = Arg->IgnoreParenCasts();
|
|
|
|
StringLiteral *Str = dyn_cast<StringLiteral>(Arg);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
if (Str == 0 || Str->isWide()) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "alias" << 1;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
// FIXME: check if target symbol exists in current file
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) AliasAttr(Attr.getLoc(), S.Context, Str->getString()));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2010-09-30 02:20:25 +08:00
|
|
|
static void HandleNakedAttr(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
// Check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isa<FunctionDecl>(d)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 0 /*function*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
d->addAttr(::new (S.Context) NakedAttr(Attr.getLoc(), S.Context));
|
|
|
|
}
|
|
|
|
|
2009-07-25 03:02:52 +08:00
|
|
|
static void HandleAlwaysInlineAttr(Decl *d, const AttributeList &Attr,
|
2008-10-28 08:17:57 +08:00
|
|
|
Sema &S) {
|
2010-09-30 02:20:25 +08:00
|
|
|
// Check the attribute arguments.
|
2008-10-28 08:17:57 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-10-28 08:17:57 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-02-20 03:16:48 +08:00
|
|
|
|
2009-04-15 01:02:11 +08:00
|
|
|
if (!isa<FunctionDecl>(d)) {
|
2009-02-20 03:16:48 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2010-09-30 02:20:25 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2009-02-20 03:16:48 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) AlwaysInlineAttr(Attr.getLoc(), S.Context));
|
2008-10-28 08:17:57 +08:00
|
|
|
}
|
|
|
|
|
2009-08-10 04:07:29 +08:00
|
|
|
static void HandleMallocAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2010-09-30 02:20:25 +08:00
|
|
|
// Check the attribute arguments.
|
2009-08-10 04:07:29 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-15 08:51:46 +08:00
|
|
|
if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(d)) {
|
2009-09-09 23:08:12 +08:00
|
|
|
QualType RetTy = FD->getResultType();
|
2009-08-15 08:51:46 +08:00
|
|
|
if (RetTy->isAnyPointerType() || RetTy->isBlockPointerType()) {
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) MallocAttr(Attr.getLoc(), S.Context));
|
2009-08-15 08:51:46 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-08-10 06:36:29 +08:00
|
|
|
}
|
|
|
|
|
2009-08-15 08:51:46 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_malloc_pointer_only);
|
2009-08-10 04:07:29 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 08:51:58 +08:00
|
|
|
static void HandleNoReturnAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
/* Diagnostics (if any) was emitted by Sema::ProcessFnAttr(). */
|
|
|
|
assert(Attr.isInvalid() == false);
|
|
|
|
d->addAttr(::new (S.Context) NoReturnAttr(Attr.getLoc(), S.Context));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleAnalyzerNoReturnAttr(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
|
|
|
|
// The checking path for 'noreturn' and 'analyzer_noreturn' are different
|
|
|
|
// because 'analyzer_noreturn' does not impact the type.
|
|
|
|
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
2010-04-30 21:10:51 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2010-08-19 08:51:58 +08:00
|
|
|
return;
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
2010-08-19 08:51:58 +08:00
|
|
|
|
2009-04-30 03:03:13 +08:00
|
|
|
if (!isFunctionOrMethod(d) && !isa<BlockDecl>(d)) {
|
|
|
|
ValueDecl *VD = dyn_cast<ValueDecl>(d);
|
2009-12-15 11:11:10 +08:00
|
|
|
if (VD == 0 || (!VD->getType()->isBlockPointerType()
|
|
|
|
&& !VD->getType()->isFunctionPointerType())) {
|
2010-04-30 21:10:51 +08:00
|
|
|
S.Diag(Attr.getLoc(),
|
|
|
|
Attr.isCXX0XAttribute() ? diag::err_attribute_wrong_decl_type
|
2010-08-19 08:51:58 +08:00
|
|
|
: diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 0 /*function*/;
|
|
|
|
return;
|
2009-04-30 03:03:13 +08:00
|
|
|
}
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
2010-08-19 08:51:58 +08:00
|
|
|
|
|
|
|
d->addAttr(::new (S.Context) AnalyzerNoReturnAttr(Attr.getLoc(), S.Context));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2010-08-10 05:53:52 +08:00
|
|
|
// PS3 PPU-specific.
|
|
|
|
static void HandleVecReturnAttr(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
/*
|
|
|
|
Returning a Vector Class in Registers
|
|
|
|
|
|
|
|
According to the PPU ABI specifications, a class with a single member of vector type is returned in
|
|
|
|
memory when used as the return value of a function. This results in inefficient code when implementing
|
|
|
|
vector classes. To return the value in a single vector register, add the vecreturn attribute to the class
|
|
|
|
definition. This attribute is also applicable to struct types.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
struct Vector
|
|
|
|
{
|
|
|
|
__vector float xyzw;
|
|
|
|
} __attribute__((vecreturn));
|
|
|
|
|
|
|
|
Vector Add(Vector lhs, Vector rhs)
|
|
|
|
{
|
|
|
|
Vector result;
|
|
|
|
result.xyzw = vec_add(lhs.xyzw, rhs.xyzw);
|
|
|
|
return result; // This will be returned in a register
|
|
|
|
}
|
|
|
|
*/
|
2010-09-18 09:12:07 +08:00
|
|
|
if (!isa<RecordDecl>(d)) {
|
2010-08-10 05:53:52 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 9 /*class*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (d->getAttr<VecReturnAttr>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_repeat_attribute) << "vecreturn";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-09-18 09:12:07 +08:00
|
|
|
RecordDecl *record = cast<RecordDecl>(d);
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
if (!isa<CXXRecordDecl>(record)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_vecreturn_only_vector_member);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cast<CXXRecordDecl>(record)->isPOD()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_vecreturn_only_pod_record);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (RecordDecl::field_iterator iter = record->field_begin(); iter != record->field_end(); iter++) {
|
|
|
|
if ((count == 1) || !iter->getType()->isVectorType()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_vecreturn_only_vector_member);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) VecReturnAttr(Attr.getLoc(), S.Context));
|
2010-08-10 05:53:52 +08:00
|
|
|
}
|
|
|
|
|
2009-11-21 16:43:09 +08:00
|
|
|
static void HandleDependencyAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
if (!isFunctionOrMethod(d) && !isa<ParmVarDecl>(d)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_decl_type)
|
2010-02-06 05:31:56 +08:00
|
|
|
<< Attr.getName() << 8 /*function, method, or parameter*/;
|
2009-11-21 16:43:09 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// FIXME: Actually store the attribute on the declaration
|
|
|
|
}
|
|
|
|
|
2008-07-25 12:39:19 +08:00
|
|
|
static void HandleUnusedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-07-25 12:39:19 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-03-31 10:47:45 +08:00
|
|
|
if (!isa<VarDecl>(d) && !isa<ObjCIvarDecl>(d) && !isFunctionOrMethod(d) &&
|
|
|
|
!isa<TypeDecl>(d)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 2 /*variable and function*/;
|
2008-07-25 12:39:19 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) UnusedAttr(Attr.getLoc(), S.Context));
|
2008-07-25 12:39:19 +08:00
|
|
|
}
|
|
|
|
|
2009-02-14 03:23:53 +08:00
|
|
|
static void HandleUsedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-02-14 03:23:53 +08:00
|
|
|
if (const VarDecl *VD = dyn_cast<VarDecl>(d)) {
|
2009-02-14 06:48:56 +08:00
|
|
|
if (VD->hasLocalStorage() || VD->hasExternalStorage()) {
|
2009-02-14 03:23:53 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "used";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if (!isFunctionOrMethod(d)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 2 /*variable and function*/;
|
2009-02-14 03:23:53 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) UsedAttr(Attr.getLoc(), S.Context));
|
2009-02-14 03:23:53 +08:00
|
|
|
}
|
|
|
|
|
2008-08-01 06:40:48 +08:00
|
|
|
static void HandleConstructorAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0 && Attr.getNumArgs() != 1) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments)
|
|
|
|
<< "0 or 1";
|
2008-08-01 06:40:48 +08:00
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
2008-08-01 06:40:48 +08:00
|
|
|
|
|
|
|
int priority = 65535; // FIXME: Do not hardcode such constants.
|
|
|
|
if (Attr.getNumArgs() > 0) {
|
|
|
|
Expr *E = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
llvm::APSInt Idx(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (E->isTypeDependent() || E->isValueDependent() ||
|
|
|
|
!E->isIntegerConstantExpr(Idx, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "constructor" << 1 << E->getSourceRange();
|
2008-08-01 06:40:48 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
priority = Idx.getZExtValue();
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-04-15 01:02:11 +08:00
|
|
|
if (!isa<FunctionDecl>(d)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2008-08-01 06:40:48 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) ConstructorAttr(Attr.getLoc(), S.Context, priority));
|
2008-08-01 06:40:48 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleDestructorAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0 && Attr.getNumArgs() != 1) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments)
|
|
|
|
<< "0 or 1";
|
2008-08-01 06:40:48 +08:00
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
2008-08-01 06:40:48 +08:00
|
|
|
|
|
|
|
int priority = 65535; // FIXME: Do not hardcode such constants.
|
|
|
|
if (Attr.getNumArgs() > 0) {
|
|
|
|
Expr *E = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
llvm::APSInt Idx(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (E->isTypeDependent() || E->isValueDependent() ||
|
|
|
|
!E->isIntegerConstantExpr(Idx, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "destructor" << 1 << E->getSourceRange();
|
2008-08-01 06:40:48 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
priority = Idx.getZExtValue();
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-08-23 06:10:48 +08:00
|
|
|
if (!isa<FunctionDecl>(d)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2008-08-01 06:40:48 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) DestructorAttr(Attr.getLoc(), S.Context, priority));
|
2008-08-01 06:40:48 +08:00
|
|
|
}
|
|
|
|
|
2008-06-29 08:43:07 +08:00
|
|
|
static void HandleDeprecatedAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2010-10-07 05:18:44 +08:00
|
|
|
int noArgs = Attr.getNumArgs();
|
|
|
|
if (noArgs > 1) {
|
|
|
|
S.Diag(Attr.getLoc(),
|
|
|
|
diag::err_attribute_wrong_number_arguments) << "0 or 1";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Handle the case where deprecated attribute has a text message.
|
|
|
|
StringLiteral *SE;
|
|
|
|
if (noArgs == 1) {
|
|
|
|
Expr *ArgExpr = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
SE = dyn_cast<StringLiteral>(ArgExpr);
|
|
|
|
if (!SE) {
|
|
|
|
S.Diag(ArgExpr->getLocStart(),
|
|
|
|
diag::err_attribute_not_string) << "deprecated";
|
|
|
|
return;
|
|
|
|
}
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
2010-10-07 05:18:44 +08:00
|
|
|
else
|
|
|
|
SE = StringLiteral::CreateEmpty(S.Context, 1);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-10-07 05:18:44 +08:00
|
|
|
d->addAttr(::new (S.Context) DeprecatedAttr(Attr.getLoc(), S.Context,
|
|
|
|
SE->getString()));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2008-12-17 09:07:27 +08:00
|
|
|
static void HandleUnavailableAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
2010-10-07 07:12:32 +08:00
|
|
|
int noArgs = Attr.getNumArgs();
|
|
|
|
if (noArgs > 1) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << "0 or 1";
|
2008-12-17 09:07:27 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-10-07 07:12:32 +08:00
|
|
|
// Handle the case where unavailable attribute has a text message.
|
|
|
|
StringLiteral *SE;
|
|
|
|
if (noArgs == 1) {
|
|
|
|
Expr *ArgExpr = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
SE = dyn_cast<StringLiteral>(ArgExpr);
|
|
|
|
if (!SE) {
|
|
|
|
S.Diag(ArgExpr->getLocStart(),
|
|
|
|
diag::err_attribute_not_string) << "unavailable";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
SE = StringLiteral::CreateEmpty(S.Context, 1);
|
|
|
|
d->addAttr(::new (S.Context) UnavailableAttr(Attr.getLoc(), S.Context,
|
|
|
|
SE->getString()));
|
2008-12-17 09:07:27 +08:00
|
|
|
}
|
|
|
|
|
2008-06-29 08:43:07 +08:00
|
|
|
static void HandleVisibilityAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 1) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-29 07:36:30 +08:00
|
|
|
Expr *Arg = static_cast<Expr*>(Attr.getArg(0));
|
2008-06-27 02:38:35 +08:00
|
|
|
Arg = Arg->IgnoreParenCasts();
|
|
|
|
StringLiteral *Str = dyn_cast<StringLiteral>(Arg);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
if (Str == 0 || Str->isWide()) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "visibility" << 1;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-01-24 02:16:35 +08:00
|
|
|
llvm::StringRef TypeStr = Str->getString();
|
2010-08-19 07:23:40 +08:00
|
|
|
VisibilityAttr::VisibilityType type;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-01-24 02:16:35 +08:00
|
|
|
if (TypeStr == "default")
|
2010-08-19 07:23:40 +08:00
|
|
|
type = VisibilityAttr::Default;
|
2010-01-24 02:16:35 +08:00
|
|
|
else if (TypeStr == "hidden")
|
2010-08-19 07:23:40 +08:00
|
|
|
type = VisibilityAttr::Hidden;
|
2010-01-24 02:16:35 +08:00
|
|
|
else if (TypeStr == "internal")
|
2010-08-19 07:23:40 +08:00
|
|
|
type = VisibilityAttr::Hidden; // FIXME
|
2010-01-24 02:16:35 +08:00
|
|
|
else if (TypeStr == "protected")
|
2010-08-19 07:23:40 +08:00
|
|
|
type = VisibilityAttr::Protected;
|
2008-06-27 02:38:35 +08:00
|
|
|
else {
|
2008-11-24 05:45:46 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_unknown_visibility) << TypeStr;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) VisibilityAttr(Attr.getLoc(), S.Context, type));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2009-02-14 16:09:34 +08:00
|
|
|
static void HandleObjCExceptionAttr(Decl *D, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-02-14 16:09:34 +08:00
|
|
|
ObjCInterfaceDecl *OCI = dyn_cast<ObjCInterfaceDecl>(D);
|
|
|
|
if (OCI == 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_requires_objc_interface);
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) ObjCExceptionAttr(Attr.getLoc(), S.Context));
|
2009-02-14 16:09:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleObjCNSObject(Decl *D, const AttributeList &Attr, Sema &S) {
|
2009-01-14 07:34:40 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
2010-05-29 02:25:28 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2009-01-14 07:34:40 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-02-14 16:09:34 +08:00
|
|
|
if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D)) {
|
2009-01-14 07:34:40 +08:00
|
|
|
QualType T = TD->getUnderlyingType();
|
|
|
|
if (!T->isPointerType() ||
|
2009-07-30 05:53:49 +08:00
|
|
|
!T->getAs<PointerType>()->getPointeeType()->isRecordType()) {
|
2009-01-14 07:34:40 +08:00
|
|
|
S.Diag(TD->getLocation(), diag::err_nsobject_attribute);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) ObjCNSObjectAttr(Attr.getLoc(), S.Context));
|
2009-01-14 07:34:40 +08:00
|
|
|
}
|
|
|
|
|
2009-07-25 03:02:52 +08:00
|
|
|
static void
|
Initial implementation of function overloading in C.
This commit adds a new attribute, "overloadable", that enables C++
function overloading in C. The attribute can only be added to function
declarations, e.g.,
int *f(int) __attribute__((overloadable));
If the "overloadable" attribute exists on a function with a given
name, *all* functions with that name (and in that scope) must have the
"overloadable" attribute. Sets of overloaded functions with the
"overloadable" attribute then follow the normal C++ rules for
overloaded functions, e.g., overloads must have different
parameter-type-lists from each other.
When calling an overloaded function in C, we follow the same
overloading rules as C++, with three extensions to the set of standard
conversions:
- A value of a given struct or union type T can be converted to the
type T. This is just the identity conversion. (In C++, this would
go through a copy constructor).
- A value of pointer type T* can be converted to a value of type U*
if T and U are compatible types. This conversion has Conversion
rank (it's considered a pointer conversion in C).
- A value of type T can be converted to a value of type U if T and U
are compatible (and are not both pointer types). This conversion
has Conversion rank (it's considered to be a new kind of
conversion unique to C, a "compatible" conversion).
Known defects (and, therefore, next steps):
1) The standard-conversion handling does not understand conversions
involving _Complex or vector extensions, so it is likely to get
these wrong. We need to add these conversions.
2) All overloadable functions with the same name will have the same
linkage name, which means we'll get a collision in the linker (if
not sooner). We'll need to mangle the names of these functions.
llvm-svn: 64336
2009-02-12 07:02:49 +08:00
|
|
|
HandleOverloadableAttr(Decl *D, const AttributeList &Attr, Sema &S) {
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
2010-05-29 02:25:28 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
Initial implementation of function overloading in C.
This commit adds a new attribute, "overloadable", that enables C++
function overloading in C. The attribute can only be added to function
declarations, e.g.,
int *f(int) __attribute__((overloadable));
If the "overloadable" attribute exists on a function with a given
name, *all* functions with that name (and in that scope) must have the
"overloadable" attribute. Sets of overloaded functions with the
"overloadable" attribute then follow the normal C++ rules for
overloaded functions, e.g., overloads must have different
parameter-type-lists from each other.
When calling an overloaded function in C, we follow the same
overloading rules as C++, with three extensions to the set of standard
conversions:
- A value of a given struct or union type T can be converted to the
type T. This is just the identity conversion. (In C++, this would
go through a copy constructor).
- A value of pointer type T* can be converted to a value of type U*
if T and U are compatible types. This conversion has Conversion
rank (it's considered a pointer conversion in C).
- A value of type T can be converted to a value of type U if T and U
are compatible (and are not both pointer types). This conversion
has Conversion rank (it's considered to be a new kind of
conversion unique to C, a "compatible" conversion).
Known defects (and, therefore, next steps):
1) The standard-conversion handling does not understand conversions
involving _Complex or vector extensions, so it is likely to get
these wrong. We need to add these conversions.
2) All overloadable functions with the same name will have the same
linkage name, which means we'll get a collision in the linker (if
not sooner). We'll need to mangle the names of these functions.
llvm-svn: 64336
2009-02-12 07:02:49 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isa<FunctionDecl>(D)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_overloadable_not_function);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) OverloadableAttr(Attr.getLoc(), S.Context));
|
Initial implementation of function overloading in C.
This commit adds a new attribute, "overloadable", that enables C++
function overloading in C. The attribute can only be added to function
declarations, e.g.,
int *f(int) __attribute__((overloadable));
If the "overloadable" attribute exists on a function with a given
name, *all* functions with that name (and in that scope) must have the
"overloadable" attribute. Sets of overloaded functions with the
"overloadable" attribute then follow the normal C++ rules for
overloaded functions, e.g., overloads must have different
parameter-type-lists from each other.
When calling an overloaded function in C, we follow the same
overloading rules as C++, with three extensions to the set of standard
conversions:
- A value of a given struct or union type T can be converted to the
type T. This is just the identity conversion. (In C++, this would
go through a copy constructor).
- A value of pointer type T* can be converted to a value of type U*
if T and U are compatible types. This conversion has Conversion
rank (it's considered a pointer conversion in C).
- A value of type T can be converted to a value of type U if T and U
are compatible (and are not both pointer types). This conversion
has Conversion rank (it's considered to be a new kind of
conversion unique to C, a "compatible" conversion).
Known defects (and, therefore, next steps):
1) The standard-conversion handling does not understand conversions
involving _Complex or vector extensions, so it is likely to get
these wrong. We need to add these conversions.
2) All overloadable functions with the same name will have the same
linkage name, which means we'll get a collision in the linker (if
not sooner). We'll need to mangle the names of these functions.
llvm-svn: 64336
2009-02-12 07:02:49 +08:00
|
|
|
}
|
|
|
|
|
2008-09-19 00:44:58 +08:00
|
|
|
static void HandleBlocksAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2009-07-25 03:02:52 +08:00
|
|
|
if (!Attr.getParameterName()) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "blocks" << 1;
|
2008-09-19 00:44:58 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-09-19 00:44:58 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2008-09-19 00:44:58 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
BlocksAttr::BlockType type;
|
2008-11-20 12:42:34 +08:00
|
|
|
if (Attr.getParameterName()->isStr("byref"))
|
2008-09-19 00:44:58 +08:00
|
|
|
type = BlocksAttr::ByRef;
|
|
|
|
else {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "blocks" << Attr.getParameterName();
|
2008-09-19 00:44:58 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) BlocksAttr(Attr.getLoc(), S.Context, type));
|
2008-09-19 00:44:58 +08:00
|
|
|
}
|
|
|
|
|
2008-10-06 02:05:59 +08:00
|
|
|
static void HandleSentinelAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() > 2) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments)
|
|
|
|
<< "0, 1 or 2";
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
|
|
|
|
2008-10-06 02:05:59 +08:00
|
|
|
int sentinel = 0;
|
|
|
|
if (Attr.getNumArgs() > 0) {
|
|
|
|
Expr *E = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
llvm::APSInt Idx(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (E->isTypeDependent() || E->isValueDependent() ||
|
|
|
|
!E->isIntegerConstantExpr(Idx, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "sentinel" << 1 << E->getSourceRange();
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
sentinel = Idx.getZExtValue();
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-10-06 02:05:59 +08:00
|
|
|
if (sentinel < 0) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_sentinel_less_than_zero)
|
|
|
|
<< E->getSourceRange();
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int nullPos = 0;
|
|
|
|
if (Attr.getNumArgs() > 1) {
|
|
|
|
Expr *E = static_cast<Expr *>(Attr.getArg(1));
|
|
|
|
llvm::APSInt Idx(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (E->isTypeDependent() || E->isValueDependent() ||
|
|
|
|
!E->isIntegerConstantExpr(Idx, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "sentinel" << 2 << E->getSourceRange();
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
nullPos = Idx.getZExtValue();
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-10-06 02:05:59 +08:00
|
|
|
if (nullPos > 1 || nullPos < 0) {
|
|
|
|
// FIXME: This error message could be improved, it would be nice
|
|
|
|
// to say what the bounds actually are.
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_sentinel_not_zero_or_one)
|
|
|
|
<< E->getSourceRange();
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(d)) {
|
2009-09-22 07:43:11 +08:00
|
|
|
const FunctionType *FT = FD->getType()->getAs<FunctionType>();
|
2009-03-18 07:03:47 +08:00
|
|
|
assert(FT && "FunctionDecl has non-function type?");
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-03-18 07:03:47 +08:00
|
|
|
if (isa<FunctionNoProtoType>(FT)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_named_arguments);
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-03-18 07:03:47 +08:00
|
|
|
if (!cast<FunctionProtoType>(FT)->isVariadic()) {
|
2009-05-16 05:18:04 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic) << 0;
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
2008-10-06 02:05:59 +08:00
|
|
|
} else if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(d)) {
|
|
|
|
if (!MD->isVariadic()) {
|
2009-05-16 05:18:04 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic) << 0;
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
2009-05-15 04:53:39 +08:00
|
|
|
}
|
|
|
|
} else if (isa<BlockDecl>(d)) {
|
2009-07-25 03:02:52 +08:00
|
|
|
// Note! BlockDecl is typeless. Variadic diagnostics will be issued by the
|
|
|
|
// caller.
|
2009-05-15 04:53:39 +08:00
|
|
|
;
|
|
|
|
} else if (const VarDecl *V = dyn_cast<VarDecl>(d)) {
|
|
|
|
QualType Ty = V->getType();
|
2009-05-16 04:33:25 +08:00
|
|
|
if (Ty->isBlockPointerType() || Ty->isFunctionPointerType()) {
|
2009-07-25 03:02:52 +08:00
|
|
|
const FunctionType *FT = Ty->isFunctionPointerType() ? getFunctionType(d)
|
2009-09-22 07:43:11 +08:00
|
|
|
: Ty->getAs<BlockPointerType>()->getPointeeType()->getAs<FunctionType>();
|
2009-05-15 04:53:39 +08:00
|
|
|
if (!cast<FunctionProtoType>(FT)->isVariadic()) {
|
2009-05-16 05:18:04 +08:00
|
|
|
int m = Ty->isFunctionPointerType() ? 0 : 1;
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_sentinel_not_variadic) << m;
|
2009-05-15 04:53:39 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-08-05 05:02:39 +08:00
|
|
|
} else {
|
2009-05-15 04:53:39 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-15 04:57:28 +08:00
|
|
|
<< Attr.getName() << 6 /*function, method or block */;
|
2009-05-15 04:53:39 +08:00
|
|
|
return;
|
|
|
|
}
|
2008-10-06 02:05:59 +08:00
|
|
|
} else {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-15 04:57:28 +08:00
|
|
|
<< Attr.getName() << 6 /*function, method or block */;
|
2008-10-06 02:05:59 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) SentinelAttr(Attr.getLoc(), S.Context, sentinel, nullPos));
|
2008-10-06 02:05:59 +08:00
|
|
|
}
|
|
|
|
|
2009-02-14 15:37:35 +08:00
|
|
|
static void HandleWarnUnusedResult(Decl *D, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-03-31 02:22:15 +08:00
|
|
|
if (!isFunction(D) && !isa<ObjCMethodDecl>(D)) {
|
2009-02-14 15:37:35 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2009-02-14 15:37:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-03-31 02:22:15 +08:00
|
|
|
if (isFunction(D) && getFunctionType(D)->getResultType()->isVoidType()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_void_function_method)
|
|
|
|
<< Attr.getName() << 0;
|
2009-12-23 07:59:52 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-03-31 02:22:15 +08:00
|
|
|
if (const ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(D))
|
|
|
|
if (MD->getResultType()->isVoidType()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_void_function_method)
|
|
|
|
<< Attr.getName() << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) WarnUnusedResultAttr(Attr.getLoc(), S.Context));
|
2009-02-14 15:37:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleWeakAttr(Decl *D, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-03-06 14:39:57 +08:00
|
|
|
|
2009-07-16 09:12:24 +08:00
|
|
|
/* weak only applies to non-static declarations */
|
2010-02-24 06:00:30 +08:00
|
|
|
if (isStaticVarOrStaticFunciton(D)) {
|
2009-07-16 09:12:24 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_weak_static) <<
|
|
|
|
dyn_cast<NamedDecl>(D)->getNameAsString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-03-06 14:39:57 +08:00
|
|
|
// TODO: could also be applied to methods?
|
|
|
|
if (!isa<FunctionDecl>(D) && !isa<VarDecl>(D)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 2 /*variable and function*/;
|
2009-03-06 14:39:57 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) WeakAttr(Attr.getLoc(), S.Context));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2009-03-06 14:39:57 +08:00
|
|
|
static void HandleWeakImportAttr(Decl *D, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
2009-03-06 14:39:57 +08:00
|
|
|
|
|
|
|
// weak_import only applies to variable & function declarations.
|
|
|
|
bool isDef = false;
|
|
|
|
if (VarDecl *VD = dyn_cast<VarDecl>(D)) {
|
|
|
|
isDef = (!VD->hasExternalStorage() || VD->getInit());
|
|
|
|
} else if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
2010-07-07 19:31:19 +08:00
|
|
|
isDef = FD->hasBody();
|
2009-05-05 03:35:12 +08:00
|
|
|
} else if (isa<ObjCPropertyDecl>(D) || isa<ObjCMethodDecl>(D)) {
|
|
|
|
// We ignore weak import on properties and methods
|
2009-03-19 01:39:31 +08:00
|
|
|
return;
|
2009-11-18 03:08:08 +08:00
|
|
|
} else if (!(S.LangOpts.ObjCNonFragileABI && isa<ObjCInterfaceDecl>(D))) {
|
2010-04-14 04:22:35 +08:00
|
|
|
// Don't issue the warning for darwin as target; yet, ignore the attribute.
|
2010-04-13 00:57:31 +08:00
|
|
|
if (S.Context.Target.getTriple().getOS() != llvm::Triple::Darwin ||
|
2010-04-14 04:22:35 +08:00
|
|
|
!isa<ObjCInterfaceDecl>(D))
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2010-04-13 00:57:31 +08:00
|
|
|
<< Attr.getName() << 2 /*variable and function*/;
|
|
|
|
return;
|
2009-03-06 14:39:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Merge should handle any subsequent violations.
|
|
|
|
if (isDef) {
|
2009-07-25 03:02:52 +08:00
|
|
|
S.Diag(Attr.getLoc(),
|
2009-03-06 14:39:57 +08:00
|
|
|
diag::warn_attribute_weak_import_invalid_on_definition)
|
|
|
|
<< "weak_import" << 2 /*variable and function*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) WeakImportAttr(Attr.getLoc(), S.Context));
|
2009-03-06 14:39:57 +08:00
|
|
|
}
|
|
|
|
|
2009-06-26 14:32:41 +08:00
|
|
|
static void HandleReqdWorkGroupSize(Decl *D, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
// Attribute has 3 arguments.
|
|
|
|
if (Attr.getNumArgs() != 3) {
|
2010-05-29 02:25:28 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2009-06-26 14:32:41 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned WGSize[3];
|
|
|
|
for (unsigned i = 0; i < 3; ++i) {
|
|
|
|
Expr *E = static_cast<Expr *>(Attr.getArg(i));
|
|
|
|
llvm::APSInt ArgNum(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (E->isTypeDependent() || E->isValueDependent() ||
|
|
|
|
!E->isIntegerConstantExpr(ArgNum, S.Context)) {
|
2009-06-26 14:32:41 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
|
|
|
|
<< "reqd_work_group_size" << E->getSourceRange();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
WGSize[i] = (unsigned) ArgNum.getZExtValue();
|
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) ReqdWorkGroupSizeAttr(Attr.getLoc(), S.Context,
|
|
|
|
WGSize[0], WGSize[1],
|
2009-06-26 14:32:41 +08:00
|
|
|
WGSize[2]));
|
|
|
|
}
|
|
|
|
|
2009-02-14 15:37:35 +08:00
|
|
|
static void HandleSectionAttr(Decl *D, const AttributeList &Attr, Sema &S) {
|
2009-02-13 01:28:23 +08:00
|
|
|
// Attribute has no arguments.
|
|
|
|
if (Attr.getNumArgs() != 1) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure that there is a string literal as the sections's single
|
|
|
|
// argument.
|
2009-08-11 03:03:04 +08:00
|
|
|
Expr *ArgExpr = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
StringLiteral *SE = dyn_cast<StringLiteral>(ArgExpr);
|
2009-02-13 01:28:23 +08:00
|
|
|
if (!SE) {
|
2009-08-11 03:03:04 +08:00
|
|
|
S.Diag(ArgExpr->getLocStart(), diag::err_attribute_not_string) << "section";
|
2009-02-13 01:28:23 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-08-11 03:03:04 +08:00
|
|
|
// If the target wants to validate the section specifier, make it happen.
|
2009-12-01 01:08:26 +08:00
|
|
|
std::string Error = S.Context.Target.isValidSectionSpecifier(SE->getString());
|
2010-01-13 04:58:53 +08:00
|
|
|
if (!Error.empty()) {
|
|
|
|
S.Diag(SE->getLocStart(), diag::err_attribute_section_invalid_for_target)
|
|
|
|
<< Error;
|
2009-08-11 03:03:04 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-01-13 04:58:53 +08:00
|
|
|
// This attribute cannot be applied to local variables.
|
|
|
|
if (isa<VarDecl>(D) && cast<VarDecl>(D)->hasLocalStorage()) {
|
|
|
|
S.Diag(SE->getLocStart(), diag::err_attribute_section_local_variable);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) SectionAttr(Attr.getLoc(), S.Context, SE->getString()));
|
2009-02-13 01:28:23 +08:00
|
|
|
}
|
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
|
2008-06-29 08:43:07 +08:00
|
|
|
static void HandleNothrowAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) NoThrowAttr(Attr.getLoc(), S.Context));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2008-10-06 07:32:53 +08:00
|
|
|
static void HandleConstAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-10-06 07:32:53 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) ConstAttr(Attr.getLoc(), S.Context));
|
2008-10-06 07:32:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void HandlePureAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-10-06 07:32:53 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) PureAttr(Attr.getLoc(), S.Context));
|
2008-10-06 07:32:53 +08:00
|
|
|
}
|
|
|
|
|
2009-01-31 09:16:18 +08:00
|
|
|
static void HandleCleanupAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2009-07-25 03:02:52 +08:00
|
|
|
if (!Attr.getParameterName()) {
|
2009-01-31 09:16:18 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-01-31 09:16:18 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-01-31 09:16:18 +08:00
|
|
|
VarDecl *VD = dyn_cast<VarDecl>(d);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-01-31 09:16:18 +08:00
|
|
|
if (!VD || !VD->hasLocalStorage()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << "cleanup";
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-01-31 09:16:18 +08:00
|
|
|
// Look up the function
|
2010-04-16 06:33:43 +08:00
|
|
|
// FIXME: Lookup probably isn't looking in the right place
|
|
|
|
// FIXME: The lookup source location should be in the attribute, not the
|
|
|
|
// start of the attribute.
|
2009-10-10 05:13:30 +08:00
|
|
|
NamedDecl *CleanupDecl
|
2010-04-16 06:33:43 +08:00
|
|
|
= S.LookupSingleName(S.TUScope, Attr.getParameterName(), Attr.getLoc(),
|
2009-10-10 05:13:30 +08:00
|
|
|
Sema::LookupOrdinaryName);
|
2009-01-31 09:16:18 +08:00
|
|
|
if (!CleanupDecl) {
|
2009-02-08 07:16:50 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_cleanup_arg_not_found) <<
|
2009-01-31 09:16:18 +08:00
|
|
|
Attr.getParameterName();
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-01-31 09:16:18 +08:00
|
|
|
FunctionDecl *FD = dyn_cast<FunctionDecl>(CleanupDecl);
|
|
|
|
if (!FD) {
|
2009-02-08 07:16:50 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_cleanup_arg_not_function) <<
|
2009-01-31 09:16:18 +08:00
|
|
|
Attr.getParameterName();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FD->getNumParams() != 1) {
|
2009-02-08 07:16:50 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_cleanup_func_must_take_one_arg) <<
|
2009-01-31 09:16:18 +08:00
|
|
|
Attr.getParameterName();
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-02-08 07:16:50 +08:00
|
|
|
// We're currently more strict than GCC about what function types we accept.
|
|
|
|
// If this ever proves to be a problem it should be easy to fix.
|
|
|
|
QualType Ty = S.Context.getPointerType(VD->getType());
|
|
|
|
QualType ParamTy = FD->getParamDecl(0)->getType();
|
2009-04-26 09:30:08 +08:00
|
|
|
if (S.CheckAssignmentConstraints(ParamTy, Ty) != Sema::Compatible) {
|
2009-07-25 03:02:52 +08:00
|
|
|
S.Diag(Attr.getLoc(),
|
2009-02-08 07:16:50 +08:00
|
|
|
diag::err_attribute_cleanup_func_arg_incompatible_type) <<
|
|
|
|
Attr.getParameterName() << ParamTy << Ty;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) CleanupAttr(Attr.getLoc(), S.Context, FD));
|
2009-01-31 09:16:18 +08:00
|
|
|
}
|
|
|
|
|
2009-07-25 03:02:52 +08:00
|
|
|
/// Handle __attribute__((format_arg((idx)))) attribute based on
|
|
|
|
/// http://gcc.gnu.org/onlinedocs/gcc/Function-Attributes.html
|
|
|
|
static void HandleFormatArgAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2009-05-21 01:41:43 +08:00
|
|
|
if (Attr.getNumArgs() != 1) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!isFunctionOrMethod(d) || !hasFunctionProto(d)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 0 /*function*/;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
// FIXME: in C++ the implicit 'this' function parameter also counts. this is
|
|
|
|
// needed in order to be compatible with GCC the index must start with 1.
|
2009-05-21 01:41:43 +08:00
|
|
|
unsigned NumArgs = getFunctionOrMethodNumArgs(d);
|
|
|
|
unsigned FirstIdx = 1;
|
|
|
|
// checks for the 2nd argument
|
|
|
|
Expr *IdxExpr = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
llvm::APSInt Idx(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (IdxExpr->isTypeDependent() || IdxExpr->isValueDependent() ||
|
|
|
|
!IdxExpr->isIntegerConstantExpr(Idx, S.Context)) {
|
2009-05-21 01:41:43 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
|
|
|
|
<< "format" << 2 << IdxExpr->getSourceRange();
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-05-21 01:41:43 +08:00
|
|
|
if (Idx.getZExtValue() < FirstIdx || Idx.getZExtValue() > NumArgs) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
|
|
|
|
<< "format" << 2 << IdxExpr->getSourceRange();
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-05-21 01:41:43 +08:00
|
|
|
unsigned ArgIdx = Idx.getZExtValue() - 1;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-05-21 01:41:43 +08:00
|
|
|
// make sure the format string is really a string
|
|
|
|
QualType Ty = getFunctionOrMethodArgType(d, ArgIdx);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-05-21 01:41:43 +08:00
|
|
|
bool not_nsstring_type = !isNSStringType(Ty, S.Context);
|
|
|
|
if (not_nsstring_type &&
|
|
|
|
!isCFStringType(Ty, S.Context) &&
|
|
|
|
(!Ty->isPointerType() ||
|
2009-07-30 05:53:49 +08:00
|
|
|
!Ty->getAs<PointerType>()->getPointeeType()->isCharType())) {
|
2009-05-21 01:41:43 +08:00
|
|
|
// FIXME: Should highlight the actual expression that has the wrong type.
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
|
2009-07-25 03:02:52 +08:00
|
|
|
<< (not_nsstring_type ? "a string type" : "an NSString")
|
2009-05-21 01:41:43 +08:00
|
|
|
<< IdxExpr->getSourceRange();
|
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
2009-05-21 01:41:43 +08:00
|
|
|
Ty = getFunctionOrMethodResultType(d);
|
|
|
|
if (!isNSStringType(Ty, S.Context) &&
|
|
|
|
!isCFStringType(Ty, S.Context) &&
|
|
|
|
(!Ty->isPointerType() ||
|
2009-07-30 05:53:49 +08:00
|
|
|
!Ty->getAs<PointerType>()->getPointeeType()->isCharType())) {
|
2009-05-21 01:41:43 +08:00
|
|
|
// FIXME: Should highlight the actual expression that has the wrong type.
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_format_attribute_result_not)
|
2009-07-25 03:02:52 +08:00
|
|
|
<< (not_nsstring_type ? "string type" : "NSString")
|
2009-05-21 01:41:43 +08:00
|
|
|
<< IdxExpr->getSourceRange();
|
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) FormatArgAttr(Attr.getLoc(), S.Context, Idx.getZExtValue()));
|
2009-05-21 01:41:43 +08:00
|
|
|
}
|
|
|
|
|
2009-10-18 10:09:17 +08:00
|
|
|
enum FormatAttrKind {
|
|
|
|
CFStringFormat,
|
|
|
|
NSStringFormat,
|
|
|
|
StrftimeFormat,
|
|
|
|
SupportedFormat,
|
2010-03-23 05:08:50 +08:00
|
|
|
IgnoredFormat,
|
2009-10-18 10:09:17 +08:00
|
|
|
InvalidFormat
|
|
|
|
};
|
|
|
|
|
|
|
|
/// getFormatAttrKind - Map from format attribute names to supported format
|
|
|
|
/// types.
|
|
|
|
static FormatAttrKind getFormatAttrKind(llvm::StringRef Format) {
|
|
|
|
// Check for formats that get handled specially.
|
|
|
|
if (Format == "NSString")
|
|
|
|
return NSStringFormat;
|
|
|
|
if (Format == "CFString")
|
|
|
|
return CFStringFormat;
|
|
|
|
if (Format == "strftime")
|
|
|
|
return StrftimeFormat;
|
|
|
|
|
|
|
|
// Otherwise, check for supported formats.
|
|
|
|
if (Format == "scanf" || Format == "printf" || Format == "printf0" ||
|
|
|
|
Format == "strfmon" || Format == "cmn_err" || Format == "strftime" ||
|
|
|
|
Format == "NSString" || Format == "CFString" || Format == "vcmn_err" ||
|
|
|
|
Format == "zcmn_err")
|
|
|
|
return SupportedFormat;
|
|
|
|
|
2010-03-23 22:44:19 +08:00
|
|
|
if (Format == "gcc_diag" || Format == "gcc_cdiag" ||
|
|
|
|
Format == "gcc_cxxdiag" || Format == "gcc_tdiag")
|
2010-03-23 05:08:50 +08:00
|
|
|
return IgnoredFormat;
|
|
|
|
|
2009-10-18 10:09:17 +08:00
|
|
|
return InvalidFormat;
|
|
|
|
}
|
|
|
|
|
2010-06-19 05:44:06 +08:00
|
|
|
/// Handle __attribute__((init_priority(priority))) attributes based on
|
|
|
|
/// http://gcc.gnu.org/onlinedocs/gcc/C_002b_002b-Attributes.html
|
|
|
|
static void HandleInitPriorityAttr(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
if (!S.getLangOptions().CPlusPlus) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_ignored) << Attr.getName();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-06-19 07:14:53 +08:00
|
|
|
if (!isa<VarDecl>(d) || S.getCurFunctionOrMethodDecl()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_init_priority_object_attr);
|
|
|
|
Attr.setInvalid();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QualType T = dyn_cast<VarDecl>(d)->getType();
|
|
|
|
if (S.Context.getAsArrayType(T))
|
|
|
|
T = S.Context.getBaseElementType(T);
|
|
|
|
if (!T->getAs<RecordType>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_init_priority_object_attr);
|
|
|
|
Attr.setInvalid();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-06-19 05:44:06 +08:00
|
|
|
if (Attr.getNumArgs() != 1) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
|
|
|
Attr.setInvalid();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Expr *priorityExpr = static_cast<Expr *>(Attr.getArg(0));
|
2010-06-19 07:14:53 +08:00
|
|
|
|
2010-06-19 05:44:06 +08:00
|
|
|
llvm::APSInt priority(32);
|
|
|
|
if (priorityExpr->isTypeDependent() || priorityExpr->isValueDependent() ||
|
|
|
|
!priorityExpr->isIntegerConstantExpr(priority, S.Context)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
|
|
|
|
<< "init_priority" << priorityExpr->getSourceRange();
|
|
|
|
Attr.setInvalid();
|
|
|
|
return;
|
|
|
|
}
|
2010-06-22 02:45:05 +08:00
|
|
|
unsigned prioritynum = priority.getZExtValue();
|
2010-06-19 05:44:06 +08:00
|
|
|
if (prioritynum < 101 || prioritynum > 65535) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_outof_range)
|
|
|
|
<< priorityExpr->getSourceRange();
|
|
|
|
Attr.setInvalid();
|
|
|
|
return;
|
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) InitPriorityAttr(Attr.getLoc(), S.Context, prioritynum));
|
2010-06-19 05:44:06 +08:00
|
|
|
}
|
|
|
|
|
2009-07-25 03:02:52 +08:00
|
|
|
/// Handle __attribute__((format(type,idx,firstarg))) attributes based on
|
|
|
|
/// http://gcc.gnu.org/onlinedocs/gcc/Function-Attributes.html
|
2008-06-29 08:43:07 +08:00
|
|
|
static void HandleFormatAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
|
2008-06-29 07:36:30 +08:00
|
|
|
if (!Attr.getParameterName()) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_string)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "format" << 1;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 2) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 3;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-05-16 07:15:03 +08:00
|
|
|
if (!isFunctionOrMethodOrBlock(d) || !hasFunctionProto(d)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-09-26 12:12:28 +08:00
|
|
|
unsigned NumArgs = getFunctionOrMethodNumArgs(d);
|
2008-06-27 02:38:35 +08:00
|
|
|
unsigned FirstIdx = 1;
|
|
|
|
|
2009-10-19 05:17:35 +08:00
|
|
|
llvm::StringRef Format = Attr.getParameterName()->getName();
|
2008-06-27 02:38:35 +08:00
|
|
|
|
|
|
|
// Normalize the argument, __foo__ becomes foo.
|
2009-10-18 10:09:17 +08:00
|
|
|
if (Format.startswith("__") && Format.endswith("__"))
|
|
|
|
Format = Format.substr(2, Format.size() - 4);
|
|
|
|
|
|
|
|
// Check for supported formats.
|
|
|
|
FormatAttrKind Kind = getFormatAttrKind(Format);
|
2010-03-23 05:08:50 +08:00
|
|
|
|
|
|
|
if (Kind == IgnoredFormat)
|
|
|
|
return;
|
|
|
|
|
2009-10-18 10:09:17 +08:00
|
|
|
if (Kind == InvalidFormat) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_type_not_supported)
|
2009-10-19 05:17:35 +08:00
|
|
|
<< "format" << Attr.getParameterName()->getName();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// checks for the 2nd argument
|
2008-06-29 07:36:30 +08:00
|
|
|
Expr *IdxExpr = static_cast<Expr *>(Attr.getArg(0));
|
2008-06-29 08:43:07 +08:00
|
|
|
llvm::APSInt Idx(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (IdxExpr->isTypeDependent() || IdxExpr->isValueDependent() ||
|
|
|
|
!IdxExpr->isIntegerConstantExpr(Idx, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "format" << 2 << IdxExpr->getSourceRange();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-08-25 22:12:34 +08:00
|
|
|
// FIXME: We should handle the implicit 'this' parameter in a more generic
|
|
|
|
// way that can be used for other arguments.
|
|
|
|
bool HasImplicitThisParam = false;
|
|
|
|
if (CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(d)) {
|
|
|
|
if (MD->isInstance()) {
|
|
|
|
HasImplicitThisParam = true;
|
|
|
|
NumArgs++;
|
|
|
|
}
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
if (Idx.getZExtValue() < FirstIdx || Idx.getZExtValue() > NumArgs) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "format" << 2 << IdxExpr->getSourceRange();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Do we need to bounds check?
|
|
|
|
unsigned ArgIdx = Idx.getZExtValue() - 1;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-11-18 02:02:24 +08:00
|
|
|
if (HasImplicitThisParam) {
|
|
|
|
if (ArgIdx == 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
|
|
|
|
<< "a string type" << IdxExpr->getSourceRange();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ArgIdx--;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
// make sure the format string is really a string
|
2008-09-26 12:12:28 +08:00
|
|
|
QualType Ty = getFunctionOrMethodArgType(d, ArgIdx);
|
2008-06-27 02:38:35 +08:00
|
|
|
|
2009-10-18 10:09:17 +08:00
|
|
|
if (Kind == CFStringFormat) {
|
2008-09-26 11:32:58 +08:00
|
|
|
if (!isCFStringType(Ty, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
|
|
|
|
<< "a CFString" << IdxExpr->getSourceRange();
|
2008-09-26 11:32:58 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-10-18 10:09:17 +08:00
|
|
|
} else if (Kind == NSStringFormat) {
|
2009-05-16 15:39:55 +08:00
|
|
|
// FIXME: do we need to check if the type is NSString*? What are the
|
|
|
|
// semantics?
|
2008-06-29 08:43:07 +08:00
|
|
|
if (!isNSStringType(Ty, S.Context)) {
|
2009-05-16 15:39:55 +08:00
|
|
|
// FIXME: Should highlight the actual expression that has the wrong type.
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
|
|
|
|
<< "an NSString" << IdxExpr->getSourceRange();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
2009-07-25 03:02:52 +08:00
|
|
|
}
|
2008-06-27 02:38:35 +08:00
|
|
|
} else if (!Ty->isPointerType() ||
|
2009-07-30 05:53:49 +08:00
|
|
|
!Ty->getAs<PointerType>()->getPointeeType()->isCharType()) {
|
2009-05-16 15:39:55 +08:00
|
|
|
// FIXME: Should highlight the actual expression that has the wrong type.
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_format_attribute_not)
|
|
|
|
<< "a string type" << IdxExpr->getSourceRange();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check the 3rd argument
|
2008-06-29 07:36:30 +08:00
|
|
|
Expr *FirstArgExpr = static_cast<Expr *>(Attr.getArg(1));
|
2008-06-29 08:43:07 +08:00
|
|
|
llvm::APSInt FirstArg(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (FirstArgExpr->isTypeDependent() || FirstArgExpr->isValueDependent() ||
|
|
|
|
!FirstArgExpr->isIntegerConstantExpr(FirstArg, S.Context)) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_n_not_int)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "format" << 3 << FirstArgExpr->getSourceRange();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if the function is variadic if the 3rd argument non-zero
|
|
|
|
if (FirstArg != 0) {
|
2008-09-26 12:12:28 +08:00
|
|
|
if (isFunctionOrMethodVariadic(d)) {
|
2008-06-27 02:38:35 +08:00
|
|
|
++NumArgs; // +1 for ...
|
|
|
|
} else {
|
2008-06-29 08:43:07 +08:00
|
|
|
S.Diag(d->getLocation(), diag::err_format_attribute_requires_variadic);
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-19 16:23:25 +08:00
|
|
|
// strftime requires FirstArg to be 0 because it doesn't read from any
|
|
|
|
// variable the input is just the current time + the format string.
|
2009-10-18 10:09:17 +08:00
|
|
|
if (Kind == StrftimeFormat) {
|
2008-06-27 02:38:35 +08:00
|
|
|
if (FirstArg != 0) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_format_strftime_third_parameter)
|
|
|
|
<< FirstArgExpr->getSourceRange();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// if 0 it disables parameter checking (to use with e.g. va_list)
|
|
|
|
} else if (FirstArg != 0 && FirstArg != NumArgs) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_out_of_bounds)
|
2008-11-19 16:23:25 +08:00
|
|
|
<< "format" << 3 << FirstArgExpr->getSourceRange();
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) FormatAttr(Attr.getLoc(), S.Context, Format,
|
|
|
|
Idx.getZExtValue(),
|
2009-10-18 10:09:17 +08:00
|
|
|
FirstArg.getZExtValue()));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2008-06-29 08:28:59 +08:00
|
|
|
static void HandleTransparentUnionAttr(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-04-30 06:16:16 +08:00
|
|
|
// Try to find the underlying union declaration.
|
|
|
|
RecordDecl *RD = 0;
|
2008-09-02 13:19:23 +08:00
|
|
|
TypedefDecl *TD = dyn_cast<TypedefDecl>(d);
|
2009-04-30 06:16:16 +08:00
|
|
|
if (TD && TD->getUnderlyingType()->isUnionType())
|
|
|
|
RD = TD->getUnderlyingType()->getAsUnionType()->getDecl();
|
|
|
|
else
|
|
|
|
RD = dyn_cast<RecordDecl>(d);
|
|
|
|
|
|
|
|
if (!RD || !RD->isUnion()) {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 1 /*union*/;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-04-30 06:16:16 +08:00
|
|
|
if (!RD->isDefinition()) {
|
2009-07-25 03:02:52 +08:00
|
|
|
S.Diag(Attr.getLoc(),
|
2009-04-30 06:16:16 +08:00
|
|
|
diag::warn_transparent_union_attribute_not_definition);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-06-30 10:36:12 +08:00
|
|
|
RecordDecl::field_iterator Field = RD->field_begin(),
|
|
|
|
FieldEnd = RD->field_end();
|
2009-04-30 06:16:16 +08:00
|
|
|
if (Field == FieldEnd) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_transparent_union_attribute_zero_fields);
|
|
|
|
return;
|
|
|
|
}
|
2008-09-02 13:19:23 +08:00
|
|
|
|
2009-04-30 06:16:16 +08:00
|
|
|
FieldDecl *FirstField = *Field;
|
|
|
|
QualType FirstType = FirstField->getType();
|
2010-07-01 01:24:13 +08:00
|
|
|
if (FirstType->hasFloatingRepresentation() || FirstType->isVectorType()) {
|
2009-07-25 03:02:52 +08:00
|
|
|
S.Diag(FirstField->getLocation(),
|
2010-07-01 01:24:13 +08:00
|
|
|
diag::warn_transparent_union_attribute_floating)
|
|
|
|
<< FirstType->isVectorType() << FirstType;
|
2009-04-30 06:16:16 +08:00
|
|
|
return;
|
|
|
|
}
|
2008-09-02 13:19:23 +08:00
|
|
|
|
2009-04-30 06:16:16 +08:00
|
|
|
uint64_t FirstSize = S.Context.getTypeSize(FirstType);
|
|
|
|
uint64_t FirstAlign = S.Context.getTypeAlign(FirstType);
|
|
|
|
for (; Field != FieldEnd; ++Field) {
|
|
|
|
QualType FieldType = Field->getType();
|
|
|
|
if (S.Context.getTypeSize(FieldType) != FirstSize ||
|
|
|
|
S.Context.getTypeAlign(FieldType) != FirstAlign) {
|
|
|
|
// Warn if we drop the attribute.
|
|
|
|
bool isSize = S.Context.getTypeSize(FieldType) != FirstSize;
|
2009-07-25 03:02:52 +08:00
|
|
|
unsigned FieldBits = isSize? S.Context.getTypeSize(FieldType)
|
2009-04-30 06:16:16 +08:00
|
|
|
: S.Context.getTypeAlign(FieldType);
|
2009-07-25 03:02:52 +08:00
|
|
|
S.Diag(Field->getLocation(),
|
2009-04-30 06:16:16 +08:00
|
|
|
diag::warn_transparent_union_attribute_field_size_align)
|
|
|
|
<< isSize << Field->getDeclName() << FieldBits;
|
|
|
|
unsigned FirstBits = isSize? FirstSize : FirstAlign;
|
2009-07-25 03:02:52 +08:00
|
|
|
S.Diag(FirstField->getLocation(),
|
2009-04-30 06:16:16 +08:00
|
|
|
diag::note_transparent_union_first_field_size_align)
|
|
|
|
<< isSize << FirstBits;
|
2008-09-02 13:19:23 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2008-06-27 02:38:35 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
RD->addAttr(::new (S.Context) TransparentUnionAttr(Attr.getLoc(), S.Context));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2008-06-29 08:28:59 +08:00
|
|
|
static void HandleAnnotateAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() != 1) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-08-11 03:03:04 +08:00
|
|
|
Expr *ArgExpr = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
StringLiteral *SE = dyn_cast<StringLiteral>(ArgExpr);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-27 02:38:35 +08:00
|
|
|
// Make sure that there is a string literal as the annotation's single
|
|
|
|
// argument.
|
|
|
|
if (!SE) {
|
2009-08-11 03:03:04 +08:00
|
|
|
S.Diag(ArgExpr->getLocStart(), diag::err_attribute_not_string) <<"annotate";
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) AnnotateAttr(Attr.getLoc(), S.Context, SE->getString()));
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
|
|
|
|
2010-06-25 11:22:07 +08:00
|
|
|
static void HandleAlignedAttr(Decl *D, const AttributeList &Attr, Sema &S) {
|
2008-06-27 02:38:35 +08:00
|
|
|
// check the attribute arguments.
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() > 1) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-11-21 16:43:09 +08:00
|
|
|
|
|
|
|
//FIXME: The C++0x version of this attribute has more limited applicabilty
|
|
|
|
// than GNU's, and should error out when it is used to specify a
|
|
|
|
// weaker alignment, rather than being silently ignored.
|
2008-06-27 02:38:35 +08:00
|
|
|
|
2008-06-29 07:36:30 +08:00
|
|
|
if (Attr.getNumArgs() == 0) {
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (S.Context) AlignedAttr(Attr.getLoc(), S.Context, true, 0));
|
2010-06-25 11:22:07 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
S.AddAlignedAttr(Attr.getLoc(), D, static_cast<Expr *>(Attr.getArg(0)));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Sema::AddAlignedAttr(SourceLocation AttrLoc, Decl *D, Expr *E) {
|
|
|
|
if (E->isTypeDependent() || E->isValueDependent()) {
|
|
|
|
// Save dependent expressions in the AST to be instantiated.
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (Context) AlignedAttr(AttrLoc, Context, true, E));
|
2008-06-27 02:38:35 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
// FIXME: Cache the number on the Attr object?
|
2008-06-29 07:50:44 +08:00
|
|
|
llvm::APSInt Alignment(32);
|
2010-06-25 11:22:07 +08:00
|
|
|
if (!E->isIntegerConstantExpr(Alignment, Context)) {
|
|
|
|
Diag(AttrLoc, diag::err_attribute_argument_not_int)
|
|
|
|
<< "aligned" << E->getSourceRange();
|
2008-06-29 07:50:44 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-02-17 07:37:57 +08:00
|
|
|
if (!llvm::isPowerOf2_64(Alignment.getZExtValue())) {
|
2010-06-25 11:22:07 +08:00
|
|
|
Diag(AttrLoc, diag::err_attribute_aligned_not_power_of_two)
|
|
|
|
<< E->getSourceRange();
|
2009-02-17 07:37:57 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
D->addAttr(::new (Context) AlignedAttr(AttrLoc, Context, true, E));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Sema::AddAlignedAttr(SourceLocation AttrLoc, Decl *D, TypeSourceInfo *TS) {
|
|
|
|
// FIXME: Cache the number on the Attr object if non-dependent?
|
|
|
|
// FIXME: Perform checking of type validity
|
|
|
|
D->addAttr(::new (Context) AlignedAttr(AttrLoc, Context, false, TS));
|
|
|
|
return;
|
2008-06-27 02:38:35 +08:00
|
|
|
}
|
2008-06-28 06:18:37 +08:00
|
|
|
|
2009-07-25 03:02:52 +08:00
|
|
|
/// HandleModeAttr - This attribute modifies the width of a decl with primitive
|
|
|
|
/// type.
|
2008-06-28 06:18:37 +08:00
|
|
|
///
|
2009-07-25 03:02:52 +08:00
|
|
|
/// Despite what would be logical, the mode attribute is a decl attribute, not a
|
|
|
|
/// type attribute: 'int ** __attribute((mode(HI))) *G;' tries to make 'G' be
|
|
|
|
/// HImode, not an intermediate pointer.
|
2008-06-29 08:28:59 +08:00
|
|
|
static void HandleModeAttr(Decl *D, const AttributeList &Attr, Sema &S) {
|
2008-06-28 06:18:37 +08:00
|
|
|
// This attribute isn't documented, but glibc uses it. It changes
|
|
|
|
// the width of an int or unsigned int to the specified size.
|
|
|
|
|
|
|
|
// Check that there aren't any arguments
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
2008-06-28 06:18:37 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
IdentifierInfo *Name = Attr.getParameterName();
|
|
|
|
if (!Name) {
|
2008-06-29 08:28:59 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_missing_parameter_name);
|
2008-06-28 06:18:37 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-10-18 10:09:24 +08:00
|
|
|
|
2009-10-19 05:17:35 +08:00
|
|
|
llvm::StringRef Str = Attr.getParameterName()->getName();
|
2008-06-28 06:18:37 +08:00
|
|
|
|
|
|
|
// Normalize the attribute name, __foo__ becomes foo.
|
2009-10-18 10:09:24 +08:00
|
|
|
if (Str.startswith("__") && Str.endswith("__"))
|
|
|
|
Str = Str.substr(2, Str.size() - 4);
|
2008-06-28 06:18:37 +08:00
|
|
|
|
|
|
|
unsigned DestWidth = 0;
|
|
|
|
bool IntegerMode = true;
|
2009-03-03 14:41:03 +08:00
|
|
|
bool ComplexMode = false;
|
2009-10-18 10:09:24 +08:00
|
|
|
switch (Str.size()) {
|
2008-06-28 06:18:37 +08:00
|
|
|
case 2:
|
2009-03-03 14:41:03 +08:00
|
|
|
switch (Str[0]) {
|
|
|
|
case 'Q': DestWidth = 8; break;
|
|
|
|
case 'H': DestWidth = 16; break;
|
|
|
|
case 'S': DestWidth = 32; break;
|
|
|
|
case 'D': DestWidth = 64; break;
|
|
|
|
case 'X': DestWidth = 96; break;
|
|
|
|
case 'T': DestWidth = 128; break;
|
|
|
|
}
|
|
|
|
if (Str[1] == 'F') {
|
|
|
|
IntegerMode = false;
|
|
|
|
} else if (Str[1] == 'C') {
|
|
|
|
IntegerMode = false;
|
|
|
|
ComplexMode = true;
|
|
|
|
} else if (Str[1] != 'I') {
|
|
|
|
DestWidth = 0;
|
|
|
|
}
|
2008-06-28 06:18:37 +08:00
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
// FIXME: glibc uses 'word' to define register_t; this is narrower than a
|
|
|
|
// pointer on PIC16 and other embedded platforms.
|
2009-10-18 10:09:24 +08:00
|
|
|
if (Str == "word")
|
2008-06-29 08:28:59 +08:00
|
|
|
DestWidth = S.Context.Target.getPointerWidth(0);
|
2009-10-18 10:09:24 +08:00
|
|
|
else if (Str == "byte")
|
2008-06-29 08:28:59 +08:00
|
|
|
DestWidth = S.Context.Target.getCharWidth();
|
2008-06-28 06:18:37 +08:00
|
|
|
break;
|
|
|
|
case 7:
|
2009-10-18 10:09:24 +08:00
|
|
|
if (Str == "pointer")
|
2008-06-29 08:28:59 +08:00
|
|
|
DestWidth = S.Context.Target.getPointerWidth(0);
|
2008-06-28 06:18:37 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType OldTy;
|
|
|
|
if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D))
|
|
|
|
OldTy = TD->getUnderlyingType();
|
|
|
|
else if (ValueDecl *VD = dyn_cast<ValueDecl>(D))
|
|
|
|
OldTy = VD->getType();
|
|
|
|
else {
|
2008-11-19 13:08:23 +08:00
|
|
|
S.Diag(D->getLocation(), diag::err_attr_wrong_decl)
|
|
|
|
<< "mode" << SourceRange(Attr.getLoc(), Attr.getLoc());
|
2008-06-28 06:18:37 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-03-03 14:41:03 +08:00
|
|
|
|
2009-09-22 07:43:11 +08:00
|
|
|
if (!OldTy->getAs<BuiltinType>() && !OldTy->isComplexType())
|
2009-03-03 14:41:03 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_mode_not_primitive);
|
|
|
|
else if (IntegerMode) {
|
2010-06-16 08:17:44 +08:00
|
|
|
if (!OldTy->isIntegralOrEnumerationType())
|
2009-03-03 14:41:03 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_mode_wrong_type);
|
|
|
|
} else if (ComplexMode) {
|
|
|
|
if (!OldTy->isComplexType())
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_mode_wrong_type);
|
|
|
|
} else {
|
|
|
|
if (!OldTy->isFloatingType())
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_mode_wrong_type);
|
|
|
|
}
|
|
|
|
|
2009-05-16 15:39:55 +08:00
|
|
|
// FIXME: Sync this with InitializePredefinedMacros; we need to match int8_t
|
|
|
|
// and friends, at least with glibc.
|
|
|
|
// FIXME: Make sure 32/64-bit integers don't get defined to types of the wrong
|
|
|
|
// width on unusual platforms.
|
2009-02-13 10:31:07 +08:00
|
|
|
// FIXME: Make sure floating-point mappings are accurate
|
|
|
|
// FIXME: Support XF and TF types
|
2008-06-28 06:18:37 +08:00
|
|
|
QualType NewTy;
|
|
|
|
switch (DestWidth) {
|
|
|
|
case 0:
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_unknown_machine_mode) << Name;
|
2008-06-28 06:18:37 +08:00
|
|
|
return;
|
|
|
|
default:
|
2008-11-19 16:23:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
|
2008-06-28 06:18:37 +08:00
|
|
|
return;
|
|
|
|
case 8:
|
2009-03-03 14:41:03 +08:00
|
|
|
if (!IntegerMode) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
|
|
|
|
return;
|
|
|
|
}
|
2008-06-28 06:18:37 +08:00
|
|
|
if (OldTy->isSignedIntegerType())
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.SignedCharTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
else
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.UnsignedCharTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
break;
|
|
|
|
case 16:
|
2009-03-03 14:41:03 +08:00
|
|
|
if (!IntegerMode) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
|
|
|
|
return;
|
|
|
|
}
|
2008-06-28 06:18:37 +08:00
|
|
|
if (OldTy->isSignedIntegerType())
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.ShortTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
else
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.UnsignedShortTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
break;
|
|
|
|
case 32:
|
|
|
|
if (!IntegerMode)
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.FloatTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
else if (OldTy->isSignedIntegerType())
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.IntTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
else
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.UnsignedIntTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
break;
|
|
|
|
case 64:
|
|
|
|
if (!IntegerMode)
|
2008-06-29 08:28:59 +08:00
|
|
|
NewTy = S.Context.DoubleTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
else if (OldTy->isSignedIntegerType())
|
2010-01-26 14:39:24 +08:00
|
|
|
if (S.Context.Target.getLongWidth() == 64)
|
|
|
|
NewTy = S.Context.LongTy;
|
|
|
|
else
|
|
|
|
NewTy = S.Context.LongLongTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
else
|
2010-01-26 14:39:24 +08:00
|
|
|
if (S.Context.Target.getLongWidth() == 64)
|
|
|
|
NewTy = S.Context.UnsignedLongTy;
|
|
|
|
else
|
|
|
|
NewTy = S.Context.UnsignedLongLongTy;
|
2008-06-28 06:18:37 +08:00
|
|
|
break;
|
2009-03-03 14:41:03 +08:00
|
|
|
case 96:
|
|
|
|
NewTy = S.Context.LongDoubleTy;
|
|
|
|
break;
|
2009-02-13 10:31:07 +08:00
|
|
|
case 128:
|
|
|
|
if (!IntegerMode) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_unsupported_machine_mode) << Name;
|
|
|
|
return;
|
|
|
|
}
|
2009-12-29 15:07:36 +08:00
|
|
|
if (OldTy->isSignedIntegerType())
|
|
|
|
NewTy = S.Context.Int128Ty;
|
|
|
|
else
|
|
|
|
NewTy = S.Context.UnsignedInt128Ty;
|
2009-03-03 14:41:03 +08:00
|
|
|
break;
|
2008-06-28 06:18:37 +08:00
|
|
|
}
|
|
|
|
|
2009-03-03 14:41:03 +08:00
|
|
|
if (ComplexMode) {
|
|
|
|
NewTy = S.Context.getComplexType(NewTy);
|
2008-06-28 06:18:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Install the new type.
|
2009-10-24 16:00:42 +08:00
|
|
|
if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D)) {
|
|
|
|
// FIXME: preserve existing source info.
|
2009-12-07 10:54:59 +08:00
|
|
|
TD->setTypeSourceInfo(S.Context.getTrivialTypeSourceInfo(NewTy));
|
2009-10-24 16:00:42 +08:00
|
|
|
} else
|
2008-06-28 06:18:37 +08:00
|
|
|
cast<ValueDecl>(D)->setType(NewTy);
|
|
|
|
}
|
2008-06-29 08:23:49 +08:00
|
|
|
|
2009-08-27 06:31:08 +08:00
|
|
|
static void HandleNoDebugAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2009-02-13 14:46:13 +08:00
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() > 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
2009-02-13 16:11:52 +08:00
|
|
|
|
2009-02-20 03:16:48 +08:00
|
|
|
if (!isFunctionOrMethod(d)) {
|
2009-02-13 14:46:13 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2009-02-13 14:46:13 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) NoDebugAttr(Attr.getLoc(), S.Context));
|
2009-02-13 14:46:13 +08:00
|
|
|
}
|
|
|
|
|
2009-08-27 06:31:08 +08:00
|
|
|
static void HandleNoInlineAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2009-02-20 03:16:48 +08:00
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-04-15 01:02:11 +08:00
|
|
|
if (!isa<FunctionDecl>(d)) {
|
2009-02-20 03:16:48 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2009-02-20 03:16:48 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) NoInlineAttr(Attr.getLoc(), S.Context));
|
2009-02-20 03:16:48 +08:00
|
|
|
}
|
|
|
|
|
2010-06-22 08:03:40 +08:00
|
|
|
static void HandleNoInstrumentFunctionAttr(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isa<FunctionDecl>(d)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 0 /*function*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) NoInstrumentFunctionAttr(Attr.getLoc(), S.Context));
|
2010-06-22 08:03:40 +08:00
|
|
|
}
|
|
|
|
|
2009-04-21 03:12:28 +08:00
|
|
|
static void HandleGNUInlineAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
2009-04-15 00:30:50 +08:00
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-04-15 01:02:11 +08:00
|
|
|
FunctionDecl *Fn = dyn_cast<FunctionDecl>(d);
|
|
|
|
if (Fn == 0) {
|
2009-04-15 00:30:50 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2009-04-15 00:30:50 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-10-28 05:01:01 +08:00
|
|
|
if (!Fn->isInlineSpecified()) {
|
2009-04-21 03:12:28 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_gnu_inline_attribute_requires_inline);
|
2009-04-15 01:02:11 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) GNUInlineAttr(Attr.getLoc(), S.Context));
|
2009-04-15 00:30:50 +08:00
|
|
|
}
|
|
|
|
|
2010-04-30 21:10:51 +08:00
|
|
|
static void HandleCallConvAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// Diagnostic is emitted elsewhere: here we store the (valid) Attr
|
|
|
|
// in the Decl node for syntactic reasoning, e.g., pretty-printing.
|
|
|
|
assert(Attr.isInvalid() == false);
|
|
|
|
|
|
|
|
switch (Attr.getKind()) {
|
|
|
|
case AttributeList::AT_fastcall:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) FastCallAttr(Attr.getLoc(), S.Context));
|
2010-04-30 21:10:51 +08:00
|
|
|
return;
|
|
|
|
case AttributeList::AT_stdcall:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) StdCallAttr(Attr.getLoc(), S.Context));
|
2010-04-30 21:10:51 +08:00
|
|
|
return;
|
2010-05-19 00:57:00 +08:00
|
|
|
case AttributeList::AT_thiscall:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) ThisCallAttr(Attr.getLoc(), S.Context));
|
2010-08-31 07:30:49 +08:00
|
|
|
return;
|
2010-04-30 21:10:51 +08:00
|
|
|
case AttributeList::AT_cdecl:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) CDeclAttr(Attr.getLoc(), S.Context));
|
2010-04-30 21:10:51 +08:00
|
|
|
return;
|
2010-09-03 09:29:35 +08:00
|
|
|
case AttributeList::AT_pascal:
|
|
|
|
d->addAttr(::new (S.Context) PascalAttr(Attr.getLoc(), S.Context));
|
|
|
|
return;
|
2010-04-30 21:10:51 +08:00
|
|
|
default:
|
|
|
|
llvm_unreachable("unexpected attribute kind");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-28 02:38:55 +08:00
|
|
|
static void HandleRegparmAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 1) {
|
2009-03-28 05:06:47 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 1;
|
2009-03-28 02:38:55 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-03-28 05:06:47 +08:00
|
|
|
|
2009-03-28 02:38:55 +08:00
|
|
|
if (!isFunctionOrMethod(d)) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type)
|
2009-05-14 05:07:32 +08:00
|
|
|
<< Attr.getName() << 0 /*function*/;
|
2009-03-28 02:38:55 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-03-28 05:06:47 +08:00
|
|
|
|
|
|
|
Expr *NumParamsExpr = static_cast<Expr *>(Attr.getArg(0));
|
|
|
|
llvm::APSInt NumParams(32);
|
2010-05-19 07:01:22 +08:00
|
|
|
if (NumParamsExpr->isTypeDependent() || NumParamsExpr->isValueDependent() ||
|
|
|
|
!NumParamsExpr->isIntegerConstantExpr(NumParams, S.Context)) {
|
2009-03-28 05:06:47 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_argument_not_int)
|
|
|
|
<< "regparm" << NumParamsExpr->getSourceRange();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-04-04 07:38:25 +08:00
|
|
|
if (S.Context.Target.getRegParmMax() == 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_regparm_wrong_platform)
|
2009-03-28 05:06:47 +08:00
|
|
|
<< NumParamsExpr->getSourceRange();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-04-04 18:27:50 +08:00
|
|
|
if (NumParams.getLimitedValue(255) > S.Context.Target.getRegParmMax()) {
|
2009-04-04 07:38:25 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_regparm_invalid_number)
|
|
|
|
<< S.Context.Target.getRegParmMax() << NumParamsExpr->getSourceRange();
|
2009-03-28 05:06:47 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) RegparmAttr(Attr.getLoc(), S.Context,
|
|
|
|
NumParams.getZExtValue()));
|
2009-03-28 02:38:55 +08:00
|
|
|
}
|
|
|
|
|
2009-11-21 16:43:09 +08:00
|
|
|
static void HandleFinalAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
// check the attribute arguments.
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isa<CXXRecordDecl>(d)
|
|
|
|
&& (!isa<CXXMethodDecl>(d) || !cast<CXXMethodDecl>(d)->isVirtual())) {
|
|
|
|
S.Diag(Attr.getLoc(),
|
|
|
|
Attr.isCXX0XAttribute() ? diag::err_attribute_wrong_decl_type
|
|
|
|
: diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 7 /*virtual method or class*/;
|
|
|
|
return;
|
|
|
|
}
|
2009-11-25 12:20:27 +08:00
|
|
|
|
|
|
|
// FIXME: Conform to C++0x redeclaration rules.
|
|
|
|
|
|
|
|
if (d->getAttr<FinalAttr>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_repeat_attribute) << "final";
|
|
|
|
return;
|
|
|
|
}
|
2009-11-21 16:43:09 +08:00
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) FinalAttr(Attr.getLoc(), S.Context));
|
2009-11-21 16:43:09 +08:00
|
|
|
}
|
|
|
|
|
2009-11-25 12:20:27 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// C++0x member checking attributes
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
static void HandleBaseCheckAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isa<CXXRecordDecl>(d)) {
|
|
|
|
S.Diag(Attr.getLoc(),
|
|
|
|
Attr.isCXX0XAttribute() ? diag::err_attribute_wrong_decl_type
|
|
|
|
: diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 9 /*class*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (d->getAttr<BaseCheckAttr>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_repeat_attribute) << "base_check";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) BaseCheckAttr(Attr.getLoc(), S.Context));
|
2009-11-25 12:20:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleHidingAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isa<RecordDecl>(d->getDeclContext())) {
|
|
|
|
// FIXME: It's not the type that's the problem
|
|
|
|
S.Diag(Attr.getLoc(),
|
|
|
|
Attr.isCXX0XAttribute() ? diag::err_attribute_wrong_decl_type
|
|
|
|
: diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 11 /*member*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Conform to C++0x redeclaration rules.
|
|
|
|
|
|
|
|
if (d->getAttr<HidingAttr>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_repeat_attribute) << "hiding";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) HidingAttr(Attr.getLoc(), S.Context));
|
2009-11-25 12:20:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void HandleOverrideAttr(Decl *d, const AttributeList &Attr, Sema &S) {
|
|
|
|
if (Attr.getNumArgs() != 0) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_attribute_wrong_number_arguments) << 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isa<CXXMethodDecl>(d) || !cast<CXXMethodDecl>(d)->isVirtual()) {
|
|
|
|
// FIXME: It's not the type that's the problem
|
|
|
|
S.Diag(Attr.getLoc(),
|
|
|
|
Attr.isCXX0XAttribute() ? diag::err_attribute_wrong_decl_type
|
|
|
|
: diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< Attr.getName() << 10 /*virtual method*/;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Conform to C++0x redeclaration rules.
|
|
|
|
|
|
|
|
if (d->getAttr<OverrideAttr>()) {
|
|
|
|
S.Diag(Attr.getLoc(), diag::err_repeat_attribute) << "override";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) OverrideAttr(Attr.getLoc(), S.Context));
|
2009-11-25 12:20:27 +08:00
|
|
|
}
|
|
|
|
|
2009-05-09 10:44:38 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Checker-specific attribute handlers.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
static void HandleNSReturnsRetainedAttr(Decl *d, const AttributeList &Attr,
|
|
|
|
Sema &S) {
|
|
|
|
|
2009-05-14 05:07:32 +08:00
|
|
|
QualType RetTy;
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-05-14 05:07:32 +08:00
|
|
|
if (ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(d))
|
|
|
|
RetTy = MD->getResultType();
|
|
|
|
else if (FunctionDecl *FD = dyn_cast<FunctionDecl>(d))
|
|
|
|
RetTy = FD->getResultType();
|
|
|
|
else {
|
2009-08-20 07:56:48 +08:00
|
|
|
SourceLocation L = Attr.getLoc();
|
|
|
|
S.Diag(d->getLocStart(), diag::warn_attribute_wrong_decl_type)
|
|
|
|
<< SourceRange(L, L) << Attr.getName() << 3 /* function or method */;
|
2009-05-09 10:44:38 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-07-30 05:53:49 +08:00
|
|
|
if (!(S.Context.isObjCNSObjectType(RetTy) || RetTy->getAs<PointerType>()
|
2009-09-22 07:43:11 +08:00
|
|
|
|| RetTy->getAs<ObjCObjectPointerType>())) {
|
2009-08-20 07:56:48 +08:00
|
|
|
SourceLocation L = Attr.getLoc();
|
|
|
|
S.Diag(d->getLocStart(), diag::warn_ns_attribute_wrong_return_type)
|
|
|
|
<< SourceRange(L, L) << Attr.getName();
|
2009-07-25 03:02:52 +08:00
|
|
|
return;
|
2009-05-14 05:07:32 +08:00
|
|
|
}
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2009-05-09 10:44:38 +08:00
|
|
|
switch (Attr.getKind()) {
|
|
|
|
default:
|
|
|
|
assert(0 && "invalid ownership attribute");
|
|
|
|
return;
|
2010-02-18 08:05:45 +08:00
|
|
|
case AttributeList::AT_cf_returns_not_retained:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) CFReturnsNotRetainedAttr(Attr.getLoc(), S.Context));
|
2010-02-18 08:05:45 +08:00
|
|
|
return;
|
|
|
|
case AttributeList::AT_ns_returns_not_retained:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) NSReturnsNotRetainedAttr(Attr.getLoc(), S.Context));
|
2010-02-18 08:05:45 +08:00
|
|
|
return;
|
2009-05-09 10:44:38 +08:00
|
|
|
case AttributeList::AT_cf_returns_retained:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) CFReturnsRetainedAttr(Attr.getLoc(), S.Context));
|
2009-05-09 10:44:38 +08:00
|
|
|
return;
|
|
|
|
case AttributeList::AT_ns_returns_retained:
|
2010-08-19 07:23:40 +08:00
|
|
|
d->addAttr(::new (S.Context) NSReturnsRetainedAttr(Attr.getLoc(), S.Context));
|
2009-05-09 10:44:38 +08:00
|
|
|
return;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2010-02-17 02:27:26 +08:00
|
|
|
static bool isKnownDeclSpecAttr(const AttributeList &Attr) {
|
|
|
|
return Attr.getKind() == AttributeList::AT_dllimport ||
|
|
|
|
Attr.getKind() == AttributeList::AT_dllexport;
|
|
|
|
}
|
|
|
|
|
2008-06-29 08:23:49 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Top Level Sema Entry Points
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2008-12-22 03:24:58 +08:00
|
|
|
/// ProcessDeclAttribute - Apply the specific attribute to the specified decl if
|
2008-06-29 08:43:07 +08:00
|
|
|
/// the attribute applies to decls. If the attribute is a type attribute, just
|
2009-11-21 16:43:09 +08:00
|
|
|
/// silently ignore it if a GNU attribute. FIXME: Applying a C++0x attribute to
|
|
|
|
/// the wrong thing is illegal (C++0x [dcl.attr.grammar]/4).
|
2009-07-25 03:02:52 +08:00
|
|
|
static void ProcessDeclAttribute(Scope *scope, Decl *D,
|
|
|
|
const AttributeList &Attr, Sema &S) {
|
2010-04-30 21:10:51 +08:00
|
|
|
if (Attr.isInvalid())
|
|
|
|
return;
|
|
|
|
|
2010-02-17 02:27:26 +08:00
|
|
|
if (Attr.isDeclspecAttribute() && !isKnownDeclSpecAttr(Attr))
|
|
|
|
// FIXME: Try to deal with other __declspec attributes!
|
2009-06-09 07:27:34 +08:00
|
|
|
return;
|
2008-06-29 08:43:07 +08:00
|
|
|
switch (Attr.getKind()) {
|
2010-02-18 11:08:58 +08:00
|
|
|
case AttributeList::AT_IBAction: HandleIBAction(D, Attr, S); break;
|
2010-05-20 01:38:06 +08:00
|
|
|
case AttributeList::AT_IBOutlet: HandleIBOutlet(D, Attr, S); break;
|
|
|
|
case AttributeList::AT_IBOutletCollection:
|
|
|
|
HandleIBOutletCollection(D, Attr, S); break;
|
2008-06-29 08:43:07 +08:00
|
|
|
case AttributeList::AT_address_space:
|
2009-02-19 01:52:36 +08:00
|
|
|
case AttributeList::AT_objc_gc:
|
2009-12-05 05:51:28 +08:00
|
|
|
case AttributeList::AT_vector_size:
|
2009-07-25 03:02:52 +08:00
|
|
|
// Ignore these, these are type attributes, handled by
|
|
|
|
// ProcessTypeAttributes.
|
2008-06-29 08:43:07 +08:00
|
|
|
break;
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_alias: HandleAliasAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_aligned: HandleAlignedAttr (D, Attr, S); break;
|
2009-07-25 03:02:52 +08:00
|
|
|
case AttributeList::AT_always_inline:
|
2008-10-28 08:17:57 +08:00
|
|
|
HandleAlwaysInlineAttr (D, Attr, S); break;
|
2009-04-10 08:01:14 +08:00
|
|
|
case AttributeList::AT_analyzer_noreturn:
|
2009-07-25 03:02:52 +08:00
|
|
|
HandleAnalyzerNoReturnAttr (D, Attr, S); break;
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_annotate: HandleAnnotateAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_base_check: HandleBaseCheckAttr (D, Attr, S); break;
|
2009-11-21 16:43:09 +08:00
|
|
|
case AttributeList::AT_carries_dependency:
|
2009-11-25 12:20:27 +08:00
|
|
|
HandleDependencyAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_constructor: HandleConstructorAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_deprecated: HandleDeprecatedAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_destructor: HandleDestructorAttr (D, Attr, S); break;
|
2008-08-01 06:40:48 +08:00
|
|
|
case AttributeList::AT_ext_vector_type:
|
2009-06-18 05:51:59 +08:00
|
|
|
HandleExtVectorTypeAttr(scope, D, Attr, S);
|
2008-08-01 06:40:48 +08:00
|
|
|
break;
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_final: HandleFinalAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_format: HandleFormatAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_format_arg: HandleFormatArgAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_gnu_inline: HandleGNUInlineAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_hiding: HandleHidingAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_mode: HandleModeAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_malloc: HandleMallocAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_nonnull: HandleNonNullAttr (D, Attr, S); break;
|
2010-07-31 09:52:11 +08:00
|
|
|
case AttributeList::AT_ownership_returns:
|
|
|
|
case AttributeList::AT_ownership_takes:
|
|
|
|
case AttributeList::AT_ownership_holds:
|
|
|
|
HandleOwnershipAttr (D, Attr, S); break;
|
2010-09-30 02:20:25 +08:00
|
|
|
case AttributeList::AT_naked: HandleNakedAttr (D, Attr, S); break;
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_noreturn: HandleNoReturnAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_nothrow: HandleNothrowAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_override: HandleOverrideAttr (D, Attr, S); break;
|
2010-08-10 05:53:52 +08:00
|
|
|
case AttributeList::AT_vecreturn: HandleVecReturnAttr (D, Attr, S); break;
|
2009-05-09 10:44:38 +08:00
|
|
|
|
|
|
|
// Checker-specific.
|
2010-02-18 08:05:45 +08:00
|
|
|
case AttributeList::AT_ns_returns_not_retained:
|
|
|
|
case AttributeList::AT_cf_returns_not_retained:
|
2009-05-09 10:44:38 +08:00
|
|
|
case AttributeList::AT_ns_returns_retained:
|
|
|
|
case AttributeList::AT_cf_returns_retained:
|
|
|
|
HandleNSReturnsRetainedAttr(D, Attr, S); break;
|
|
|
|
|
2009-06-26 14:32:41 +08:00
|
|
|
case AttributeList::AT_reqd_wg_size:
|
|
|
|
HandleReqdWorkGroupSize(D, Attr, S); break;
|
|
|
|
|
2010-06-19 05:44:06 +08:00
|
|
|
case AttributeList::AT_init_priority:
|
|
|
|
HandleInitPriorityAttr(D, Attr, S); break;
|
|
|
|
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_packed: HandlePackedAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_section: HandleSectionAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_unavailable: HandleUnavailableAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_unused: HandleUnusedAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_used: HandleUsedAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_visibility: HandleVisibilityAttr (D, Attr, S); break;
|
2009-02-14 15:37:35 +08:00
|
|
|
case AttributeList::AT_warn_unused_result: HandleWarnUnusedResult(D,Attr,S);
|
|
|
|
break;
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_weak: HandleWeakAttr (D, Attr, S); break;
|
2010-02-24 06:00:30 +08:00
|
|
|
case AttributeList::AT_weakref: HandleWeakRefAttr (D, Attr, S); break;
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_weak_import: HandleWeakImportAttr (D, Attr, S); break;
|
2008-06-29 08:43:07 +08:00
|
|
|
case AttributeList::AT_transparent_union:
|
|
|
|
HandleTransparentUnionAttr(D, Attr, S);
|
|
|
|
break;
|
2009-02-14 16:09:34 +08:00
|
|
|
case AttributeList::AT_objc_exception:
|
|
|
|
HandleObjCExceptionAttr(D, Attr, S);
|
|
|
|
break;
|
Initial implementation of function overloading in C.
This commit adds a new attribute, "overloadable", that enables C++
function overloading in C. The attribute can only be added to function
declarations, e.g.,
int *f(int) __attribute__((overloadable));
If the "overloadable" attribute exists on a function with a given
name, *all* functions with that name (and in that scope) must have the
"overloadable" attribute. Sets of overloaded functions with the
"overloadable" attribute then follow the normal C++ rules for
overloaded functions, e.g., overloads must have different
parameter-type-lists from each other.
When calling an overloaded function in C, we follow the same
overloading rules as C++, with three extensions to the set of standard
conversions:
- A value of a given struct or union type T can be converted to the
type T. This is just the identity conversion. (In C++, this would
go through a copy constructor).
- A value of pointer type T* can be converted to a value of type U*
if T and U are compatible types. This conversion has Conversion
rank (it's considered a pointer conversion in C).
- A value of type T can be converted to a value of type U if T and U
are compatible (and are not both pointer types). This conversion
has Conversion rank (it's considered to be a new kind of
conversion unique to C, a "compatible" conversion).
Known defects (and, therefore, next steps):
1) The standard-conversion handling does not understand conversions
involving _Complex or vector extensions, so it is likely to get
these wrong. We need to add these conversions.
2) All overloadable functions with the same name will have the same
linkage name, which means we'll get a collision in the linker (if
not sooner). We'll need to mangle the names of these functions.
llvm-svn: 64336
2009-02-12 07:02:49 +08:00
|
|
|
case AttributeList::AT_overloadable:HandleOverloadableAttr(D, Attr, S); break;
|
2009-11-25 12:20:27 +08:00
|
|
|
case AttributeList::AT_nsobject: HandleObjCNSObject (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_blocks: HandleBlocksAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_sentinel: HandleSentinelAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_const: HandleConstAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_pure: HandlePureAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_cleanup: HandleCleanupAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_nodebug: HandleNoDebugAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_noinline: HandleNoInlineAttr (D, Attr, S); break;
|
|
|
|
case AttributeList::AT_regparm: HandleRegparmAttr (D, Attr, S); break;
|
2009-07-25 03:02:52 +08:00
|
|
|
case AttributeList::IgnoredAttribute:
|
2009-02-13 16:16:43 +08:00
|
|
|
// Just ignore
|
|
|
|
break;
|
2010-06-22 08:03:40 +08:00
|
|
|
case AttributeList::AT_no_instrument_function: // Interacts with -pg.
|
|
|
|
HandleNoInstrumentFunctionAttr(D, Attr, S);
|
|
|
|
break;
|
2010-02-06 05:31:56 +08:00
|
|
|
case AttributeList::AT_stdcall:
|
|
|
|
case AttributeList::AT_cdecl:
|
|
|
|
case AttributeList::AT_fastcall:
|
2010-05-19 00:57:00 +08:00
|
|
|
case AttributeList::AT_thiscall:
|
2010-09-03 09:29:35 +08:00
|
|
|
case AttributeList::AT_pascal:
|
2010-04-30 21:10:51 +08:00
|
|
|
HandleCallConvAttr(D, Attr, S);
|
2010-02-06 05:31:56 +08:00
|
|
|
break;
|
2008-06-29 08:43:07 +08:00
|
|
|
default:
|
2010-01-10 20:58:08 +08:00
|
|
|
// Ask target about the attribute.
|
|
|
|
const TargetAttributesSema &TargetAttrs = S.getTargetAttributesSema();
|
|
|
|
if (!TargetAttrs.ProcessDeclAttribute(scope, D, Attr, S))
|
2010-07-08 17:42:26 +08:00
|
|
|
S.Diag(Attr.getLoc(), diag::warn_unknown_attribute_ignored)
|
|
|
|
<< Attr.getName();
|
2008-06-29 08:43:07 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// ProcessDeclAttributeList - Apply all the decl attributes in the specified
|
|
|
|
/// attribute list to the specified decl, ignoring any type attributes.
|
2009-06-18 05:51:59 +08:00
|
|
|
void Sema::ProcessDeclAttributeList(Scope *S, Decl *D, const AttributeList *AttrList) {
|
2010-02-24 06:00:30 +08:00
|
|
|
for (const AttributeList* l = AttrList; l; l = l->getNext()) {
|
|
|
|
ProcessDeclAttribute(S, D, *l, *this);
|
|
|
|
}
|
|
|
|
|
|
|
|
// GCC accepts
|
|
|
|
// static int a9 __attribute__((weakref));
|
|
|
|
// but that looks really pointless. We reject it.
|
|
|
|
if (D->hasAttr<WeakRefAttr>() && !D->hasAttr<AliasAttr>()) {
|
|
|
|
Diag(AttrList->getLoc(), diag::err_attribute_weakref_without_alias) <<
|
2010-07-31 09:52:11 +08:00
|
|
|
dyn_cast<NamedDecl>(D)->getNameAsString();
|
2010-02-24 06:00:30 +08:00
|
|
|
return;
|
2008-06-29 08:43:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-30 11:15:39 +08:00
|
|
|
/// DeclClonePragmaWeak - clone existing decl (maybe definition),
|
|
|
|
/// #pragma weak needs a non-definition decl and source may not have one
|
2009-09-09 23:08:12 +08:00
|
|
|
NamedDecl * Sema::DeclClonePragmaWeak(NamedDecl *ND, IdentifierInfo *II) {
|
2009-07-31 10:52:19 +08:00
|
|
|
assert(isa<FunctionDecl>(ND) || isa<VarDecl>(ND));
|
2009-07-30 11:15:39 +08:00
|
|
|
NamedDecl *NewD = 0;
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(ND)) {
|
|
|
|
NewD = FunctionDecl::Create(FD->getASTContext(), FD->getDeclContext(),
|
|
|
|
FD->getLocation(), DeclarationName(II),
|
2009-12-07 10:54:59 +08:00
|
|
|
FD->getType(), FD->getTypeSourceInfo());
|
2010-03-15 18:12:16 +08:00
|
|
|
if (FD->getQualifier()) {
|
|
|
|
FunctionDecl *NewFD = cast<FunctionDecl>(NewD);
|
|
|
|
NewFD->setQualifierInfo(FD->getQualifier(), FD->getQualifierRange());
|
|
|
|
}
|
2009-07-30 11:15:39 +08:00
|
|
|
} else if (VarDecl *VD = dyn_cast<VarDecl>(ND)) {
|
|
|
|
NewD = VarDecl::Create(VD->getASTContext(), VD->getDeclContext(),
|
|
|
|
VD->getLocation(), II,
|
2009-12-07 10:54:59 +08:00
|
|
|
VD->getType(), VD->getTypeSourceInfo(),
|
2010-04-20 06:54:31 +08:00
|
|
|
VD->getStorageClass(),
|
|
|
|
VD->getStorageClassAsWritten());
|
2010-03-15 18:12:16 +08:00
|
|
|
if (VD->getQualifier()) {
|
|
|
|
VarDecl *NewVD = cast<VarDecl>(NewD);
|
|
|
|
NewVD->setQualifierInfo(VD->getQualifier(), VD->getQualifierRange());
|
|
|
|
}
|
2009-07-30 11:15:39 +08:00
|
|
|
}
|
|
|
|
return NewD;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// DeclApplyPragmaWeak - A declaration (maybe definition) needs #pragma weak
|
|
|
|
/// applied to it, possibly with an alias.
|
2009-07-31 10:52:19 +08:00
|
|
|
void Sema::DeclApplyPragmaWeak(Scope *S, NamedDecl *ND, WeakInfo &W) {
|
2009-09-09 02:10:11 +08:00
|
|
|
if (W.getUsed()) return; // only do this once
|
|
|
|
W.setUsed(true);
|
|
|
|
if (W.getAlias()) { // clone decl, impersonate __attribute(weak,alias(...))
|
|
|
|
IdentifierInfo *NDId = ND->getIdentifier();
|
|
|
|
NamedDecl *NewD = DeclClonePragmaWeak(ND, W.getAlias());
|
2010-08-19 07:23:40 +08:00
|
|
|
NewD->addAttr(::new (Context) AliasAttr(W.getLocation(), Context,
|
|
|
|
NDId->getName()));
|
|
|
|
NewD->addAttr(::new (Context) WeakAttr(W.getLocation(), Context));
|
2009-09-09 02:10:11 +08:00
|
|
|
WeakTopLevelDecl.push_back(NewD);
|
|
|
|
// FIXME: "hideous" code from Sema::LazilyCreateBuiltin
|
|
|
|
// to insert Decl at TU scope, sorry.
|
|
|
|
DeclContext *SavedContext = CurContext;
|
|
|
|
CurContext = Context.getTranslationUnitDecl();
|
|
|
|
PushOnScopeChains(NewD, S);
|
|
|
|
CurContext = SavedContext;
|
|
|
|
} else { // just add weak to existing
|
2010-08-19 07:23:40 +08:00
|
|
|
ND->addAttr(::new (Context) WeakAttr(W.getLocation(), Context));
|
2009-07-30 11:15:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-29 08:23:49 +08:00
|
|
|
/// ProcessDeclAttributes - Given a declarator (PD) with attributes indicated in
|
|
|
|
/// it, apply them to D. This is a bit tricky because PD can have attributes
|
|
|
|
/// specified in many different places, and we need to find and apply them all.
|
2009-06-18 05:51:59 +08:00
|
|
|
void Sema::ProcessDeclAttributes(Scope *S, Decl *D, const Declarator &PD) {
|
2009-07-30 11:15:39 +08:00
|
|
|
// Handle #pragma weak
|
|
|
|
if (NamedDecl *ND = dyn_cast<NamedDecl>(D)) {
|
|
|
|
if (ND->hasLinkage()) {
|
|
|
|
WeakInfo W = WeakUndeclaredIdentifiers.lookup(ND->getIdentifier());
|
|
|
|
if (W != WeakInfo()) {
|
2009-07-31 10:52:19 +08:00
|
|
|
// Identifier referenced by #pragma weak before it was declared
|
|
|
|
DeclApplyPragmaWeak(S, ND, W);
|
2009-07-30 11:15:39 +08:00
|
|
|
WeakUndeclaredIdentifiers[ND->getIdentifier()] = W;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-29 08:23:49 +08:00
|
|
|
// Apply decl attributes from the DeclSpec if present.
|
|
|
|
if (const AttributeList *Attrs = PD.getDeclSpec().getAttributes())
|
2009-06-18 05:51:59 +08:00
|
|
|
ProcessDeclAttributeList(S, D, Attrs);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-29 08:23:49 +08:00
|
|
|
// Walk the declarator structure, applying decl attributes that were in a type
|
|
|
|
// position to the decl itself. This handles cases like:
|
|
|
|
// int *__attr__(x)** D;
|
|
|
|
// when X is a decl attribute.
|
|
|
|
for (unsigned i = 0, e = PD.getNumTypeObjects(); i != e; ++i)
|
|
|
|
if (const AttributeList *Attrs = PD.getTypeObject(i).getAttrs())
|
2009-06-18 05:51:59 +08:00
|
|
|
ProcessDeclAttributeList(S, D, Attrs);
|
2009-07-25 03:02:52 +08:00
|
|
|
|
2008-06-29 08:23:49 +08:00
|
|
|
// Finally, apply any attributes on the decl itself.
|
|
|
|
if (const AttributeList *Attrs = PD.getAttributes())
|
2009-06-18 05:51:59 +08:00
|
|
|
ProcessDeclAttributeList(S, D, Attrs);
|
2008-06-29 08:23:49 +08:00
|
|
|
}
|
2009-11-04 10:18:39 +08:00
|
|
|
|
|
|
|
/// PushParsingDeclaration - Enter a new "scope" of deprecation
|
|
|
|
/// warnings.
|
|
|
|
///
|
|
|
|
/// The state token we use is the start index of this scope
|
|
|
|
/// on the warning stack.
|
2010-08-27 07:41:50 +08:00
|
|
|
Sema::ParsingDeclStackState Sema::PushParsingDeclaration() {
|
2009-11-04 10:18:39 +08:00
|
|
|
ParsingDeclDepth++;
|
2010-01-27 11:50:35 +08:00
|
|
|
return (ParsingDeclStackState) DelayedDiagnostics.size();
|
2009-11-04 10:18:39 +08:00
|
|
|
}
|
|
|
|
|
2010-08-21 17:40:31 +08:00
|
|
|
void Sema::PopParsingDeclaration(ParsingDeclStackState S, Decl *D) {
|
2009-11-04 10:18:39 +08:00
|
|
|
assert(ParsingDeclDepth > 0 && "empty ParsingDeclaration stack");
|
|
|
|
ParsingDeclDepth--;
|
|
|
|
|
2010-01-27 11:50:35 +08:00
|
|
|
if (DelayedDiagnostics.empty())
|
2009-11-04 10:18:39 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
unsigned SavedIndex = (unsigned) S;
|
2010-01-27 11:50:35 +08:00
|
|
|
assert(SavedIndex <= DelayedDiagnostics.size() &&
|
2009-11-04 10:18:39 +08:00
|
|
|
"saved index is out of bounds");
|
|
|
|
|
2010-03-16 13:22:47 +08:00
|
|
|
unsigned E = DelayedDiagnostics.size();
|
|
|
|
|
2010-01-27 11:50:35 +08:00
|
|
|
// We only want to actually emit delayed diagnostics when we
|
|
|
|
// successfully parsed a decl.
|
|
|
|
if (D) {
|
|
|
|
// We really do want to start with 0 here. We get one push for a
|
|
|
|
// decl spec and another for each declarator; in a decl group like:
|
|
|
|
// deprecated_typedef foo, *bar, baz();
|
|
|
|
// only the declarator pops will be passed decls. This is correct;
|
|
|
|
// we really do need to consider delayed diagnostics from the decl spec
|
|
|
|
// for each of the different declarations.
|
2010-03-16 13:22:47 +08:00
|
|
|
for (unsigned I = 0; I != E; ++I) {
|
2010-01-27 11:50:35 +08:00
|
|
|
if (DelayedDiagnostics[I].Triggered)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
switch (DelayedDiagnostics[I].Kind) {
|
|
|
|
case DelayedDiagnostic::Deprecation:
|
|
|
|
HandleDelayedDeprecationCheck(DelayedDiagnostics[I], D);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DelayedDiagnostic::Access:
|
|
|
|
HandleDelayedAccessCheck(DelayedDiagnostics[I], D);
|
|
|
|
break;
|
2009-11-04 10:18:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-16 13:22:47 +08:00
|
|
|
// Destroy all the delayed diagnostics we're about to pop off.
|
|
|
|
for (unsigned I = SavedIndex; I != E; ++I)
|
|
|
|
DelayedDiagnostics[I].destroy();
|
|
|
|
|
2010-01-27 11:50:35 +08:00
|
|
|
DelayedDiagnostics.set_size(SavedIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool isDeclDeprecated(Decl *D) {
|
|
|
|
do {
|
|
|
|
if (D->hasAttr<DeprecatedAttr>())
|
|
|
|
return true;
|
|
|
|
} while ((D = cast_or_null<Decl>(D->getDeclContext())));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-08-26 10:13:20 +08:00
|
|
|
void Sema::HandleDelayedDeprecationCheck(DelayedDiagnostic &DD,
|
2010-01-27 11:50:35 +08:00
|
|
|
Decl *Ctx) {
|
|
|
|
if (isDeclDeprecated(Ctx))
|
|
|
|
return;
|
|
|
|
|
|
|
|
DD.Triggered = true;
|
2010-10-09 23:49:00 +08:00
|
|
|
if (!DD.getDeprecationMessage().empty())
|
2010-10-07 05:18:44 +08:00
|
|
|
Diag(DD.Loc, diag::warn_deprecated_message)
|
2010-10-09 23:49:00 +08:00
|
|
|
<< DD.getDeprecationDecl()->getDeclName()
|
|
|
|
<< DD.getDeprecationMessage();
|
2010-10-07 05:18:44 +08:00
|
|
|
else
|
|
|
|
Diag(DD.Loc, diag::warn_deprecated)
|
2010-10-09 23:49:00 +08:00
|
|
|
<< DD.getDeprecationDecl()->getDeclName();
|
2009-11-04 10:18:39 +08:00
|
|
|
}
|
|
|
|
|
2010-10-09 23:49:00 +08:00
|
|
|
void Sema::EmitDeprecationWarning(NamedDecl *D, llvm::StringRef Message,
|
2010-10-07 05:18:44 +08:00
|
|
|
SourceLocation Loc) {
|
2009-11-04 10:18:39 +08:00
|
|
|
// Delay if we're currently parsing a declaration.
|
|
|
|
if (ParsingDeclDepth) {
|
2010-10-07 05:18:44 +08:00
|
|
|
DelayedDiagnostics.push_back(DelayedDiagnostic::makeDeprecation(Loc, D,
|
|
|
|
Message));
|
2009-11-04 10:18:39 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, don't warn if our current context is deprecated.
|
|
|
|
if (isDeclDeprecated(cast<Decl>(CurContext)))
|
|
|
|
return;
|
2010-10-09 23:49:00 +08:00
|
|
|
if (!Message.empty())
|
2010-10-07 05:18:44 +08:00
|
|
|
Diag(Loc, diag::warn_deprecated_message) << D->getDeclName()
|
|
|
|
<< Message;
|
|
|
|
else
|
|
|
|
Diag(Loc, diag::warn_deprecated) << D->getDeclName();
|
2009-11-04 10:18:39 +08:00
|
|
|
}
|