Some early declarations to support sentinel attribute on

message dispatches (and function calls later). No change in
functionality.

llvm-svn: 71683
This commit is contained in:
Fariborz Jahanian 2009-05-13 18:09:35 +00:00
parent 784ae669ab
commit 027b886a99
6 changed files with 45 additions and 0 deletions

View File

@ -66,6 +66,7 @@ public:
Pure, Pure,
Regparm, Regparm,
Section, Section,
Sentinel,
StdCall, StdCall,
TransparentUnion, TransparentUnion,
Unavailable, Unavailable,
@ -359,6 +360,23 @@ public:
static bool classof(const FormatAttr *A) { return true; } static bool classof(const FormatAttr *A) { return true; }
}; };
class SentinelAttr : public Attr {
int sentinel, NullPos;
public:
SentinelAttr(int sentinel_val, int nullPos) : Attr(Sentinel),
sentinel(sentinel_val), NullPos(nullPos) {}
int getSentinel() const { return sentinel; }
int getNullPos() const { return NullPos; }
virtual Attr *clone(ASTContext &C) const {
return ::new (C) SentinelAttr(sentinel, NullPos);
}
// Implement isa/cast/dyncast/etc.
static bool classof(const Attr *A) { return A->getKind() == Sentinel; }
static bool classof(const SentinelAttr *A) { return true; }
};
class VisibilityAttr : public Attr { class VisibilityAttr : public Attr {
public: public:
/// @brief An enumeration for the kinds of visibility of symbols. /// @brief An enumeration for the kinds of visibility of symbols.

View File

@ -452,6 +452,13 @@ Attr *PCHReader::ReadAttributes() {
New = ::new (*Context) FormatAttr(Type, FormatIdx, FirstArg); New = ::new (*Context) FormatAttr(Type, FormatIdx, FirstArg);
break; break;
} }
case Attr::Sentinel: {
int sentinel = Record[Idx++];
int nullPos = Record[Idx++];
New = ::new (*Context) SentinelAttr(sentinel, nullPos);
break;
}
SIMPLE_ATTR(GNUInline); SIMPLE_ATTR(GNUInline);

View File

@ -1556,6 +1556,13 @@ void PCHWriter::WriteAttributeRecord(const Attr *Attr) {
break; break;
} }
case Attr::Sentinel : {
const SentinelAttr *Sentinel = cast<SentinelAttr>(Attr);
Record.push_back(Sentinel->getSentinel());
Record.push_back(Sentinel->getNullPos());
break;
}
case Attr::GNUInline: case Attr::GNUInline:
case Attr::IBOutletKind: case Attr::IBOutletKind:
case Attr::NoReturn: case Attr::NoReturn:

View File

@ -1274,6 +1274,8 @@ public:
bool DiagnosePropertyAccessorMismatch(ObjCPropertyDecl *PD, bool DiagnosePropertyAccessorMismatch(ObjCPropertyDecl *PD,
ObjCMethodDecl *Getter, ObjCMethodDecl *Getter,
SourceLocation Loc); SourceLocation Loc);
void DiagnoseSentinelCalls(NamedDecl *D, SourceLocation Loc,
Expr **Args, unsigned NumArgs);
// Primary Expressions. // Primary Expressions.
virtual SourceRange getExprRange(ExprTy *E) const; virtual SourceRange getExprRange(ExprTy *E) const;

View File

@ -88,6 +88,15 @@ bool Sema::DiagnoseUseOfDecl(NamedDecl *D, SourceLocation Loc) {
return false; return false;
} }
/// DiagnoseSentinelCalls - This routine checks on method dispatch calls
/// (and other functions in future), which have been declared with sentinel
/// attribute. It warns if call does not have the sentinel argument.
///
void Sema::DiagnoseSentinelCalls(NamedDecl *D, SourceLocation Loc,
Expr **Args, unsigned NumArgs)
{
}
SourceRange Sema::getExprRange(ExprTy *E) const { SourceRange Sema::getExprRange(ExprTy *E) const {
Expr *Ex = (Expr *)E; Expr *Ex = (Expr *)E;
return Ex? Ex->getSourceRange() : SourceRange(); return Ex? Ex->getSourceRange() : SourceRange();

View File

@ -626,6 +626,8 @@ Sema::ExprResult Sema::ActOnInstanceMessage(ExprTy *receiver, Selector Sel,
return true; return true;
} }
if (Method)
DiagnoseSentinelCalls(Method, receiverLoc, ArgExprs, NumArgs);
if (CheckMessageArgumentTypes(ArgExprs, NumArgs, Sel, Method, false, if (CheckMessageArgumentTypes(ArgExprs, NumArgs, Sel, Method, false,
lbrac, rbrac, returnType)) lbrac, rbrac, returnType))
return true; return true;