forked from OSchip/llvm-project
Diagnose a variety of access of ivars when they conflict with
local or global variables in instance/class methods. llvm-svn: 65879
This commit is contained in:
parent
53b72b4fea
commit
bf8e842b67
|
@ -1436,4 +1436,8 @@ DIAG(err_invalid_protocol_qualifiers, ERROR,
|
||||||
"invalid protocol qualifiers on non-ObjC type")
|
"invalid protocol qualifiers on non-ObjC type")
|
||||||
DIAG(err_qualified_class_unsupported, ERROR,
|
DIAG(err_qualified_class_unsupported, ERROR,
|
||||||
"protocol qualified 'Class' is unsupported")
|
"protocol qualified 'Class' is unsupported")
|
||||||
|
DIAG(warn_ivar_use_hidden, WARNING,
|
||||||
|
"local declaration of %0 hides instance variable")
|
||||||
|
DIAG(error_ivar_use_in_class_method, ERROR,
|
||||||
|
"instance variable %0 accessed in class method")
|
||||||
|
|
||||||
|
|
|
@ -627,27 +627,42 @@ Sema::ActOnDeclarationNameExpr(Scope *S, SourceLocation Loc,
|
||||||
if (II && getCurMethodDecl()) {
|
if (II && getCurMethodDecl()) {
|
||||||
// There are two cases to handle here. 1) scoped lookup could have failed,
|
// There are two cases to handle here. 1) scoped lookup could have failed,
|
||||||
// in which case we should look for an ivar. 2) scoped lookup could have
|
// in which case we should look for an ivar. 2) scoped lookup could have
|
||||||
// found a decl, but that decl is outside the current method (i.e. a global
|
// found a decl, but that decl is outside the current instance method (i.e.
|
||||||
// variable). In these two cases, we do a lookup for an ivar with this
|
// a global variable). In these two cases, we do a lookup for an ivar with
|
||||||
// name, if the lookup suceeds, we replace it our current decl.
|
// this name, if the lookup sucedes, we replace it our current decl.
|
||||||
if (D == 0 || D->isDefinedOutsideFunctionOrMethod()) {
|
if (D == 0 || D->isDefinedOutsideFunctionOrMethod()) {
|
||||||
ObjCInterfaceDecl *IFace = getCurMethodDecl()->getClassInterface();
|
ObjCInterfaceDecl *IFace = getCurMethodDecl()->getClassInterface();
|
||||||
if (ObjCIvarDecl *IV = IFace->lookupInstanceVariable(II)) {
|
if (ObjCIvarDecl *IV = IFace->lookupInstanceVariable(II)) {
|
||||||
// Check if referencing a field with __attribute__((deprecated)).
|
// Check if referencing a field with __attribute__((deprecated)).
|
||||||
if (DiagnoseUseOfDecl(IV, Loc))
|
if (DiagnoseUseOfDecl(IV, Loc))
|
||||||
return ExprError();
|
return ExprError();
|
||||||
|
bool IsClsMethod = getCurMethodDecl()->isClassMethod();
|
||||||
// FIXME: This should use a new expr for a direct reference, don't turn
|
// If a class method attemps to use a free standing ivar, this is
|
||||||
// this into Self->ivar, just return a BareIVarExpr or something.
|
// an error.
|
||||||
IdentifierInfo &II = Context.Idents.get("self");
|
if (IsClsMethod && D && !D->isDefinedOutsideFunctionOrMethod())
|
||||||
OwningExprResult SelfExpr = ActOnIdentifierExpr(S, Loc, II, false);
|
return ExprError(Diag(Loc, diag::error_ivar_use_in_class_method)
|
||||||
ObjCIvarRefExpr *MRef = new (Context) ObjCIvarRefExpr(IV, IV->getType(),
|
<< IV->getDeclName());
|
||||||
Loc, static_cast<Expr*>(SelfExpr.release()),
|
// If a class method uses a global variable, even if an ivar with
|
||||||
true, true);
|
// same name exists, use the global.
|
||||||
Context.setFieldDecl(IFace, IV, MRef);
|
if (!IsClsMethod) {
|
||||||
return Owned(MRef);
|
// FIXME: This should use a new expr for a direct reference, don't turn
|
||||||
|
// this into Self->ivar, just return a BareIVarExpr or something.
|
||||||
|
IdentifierInfo &II = Context.Idents.get("self");
|
||||||
|
OwningExprResult SelfExpr = ActOnIdentifierExpr(S, Loc, II, false);
|
||||||
|
ObjCIvarRefExpr *MRef = new (Context) ObjCIvarRefExpr(IV, IV->getType(),
|
||||||
|
Loc, static_cast<Expr*>(SelfExpr.release()),
|
||||||
|
true, true);
|
||||||
|
Context.setFieldDecl(IFace, IV, MRef);
|
||||||
|
return Owned(MRef);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else if (getCurMethodDecl()->isInstanceMethod()) {
|
||||||
|
// We should warn if a local variable hides an ivar.
|
||||||
|
ObjCInterfaceDecl *IFace = getCurMethodDecl()->getClassInterface();
|
||||||
|
if (ObjCIvarDecl *IV = IFace->lookupInstanceVariable(II))
|
||||||
|
Diag(Loc, diag::warn_ivar_use_hidden)<<IV->getDeclName();
|
||||||
|
}
|
||||||
// Needed to implement property "super.method" notation.
|
// Needed to implement property "super.method" notation.
|
||||||
if (D == 0 && II->isStr("super")) {
|
if (D == 0 && II->isStr("super")) {
|
||||||
QualType T = Context.getPointerType(Context.getObjCInterfaceType(
|
QualType T = Context.getPointerType(Context.getObjCInterfaceType(
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
// RUN: clang -fsyntax-only -verify %s
|
||||||
|
|
||||||
|
@interface Sprite {
|
||||||
|
int sprite, spree;
|
||||||
|
int UseGlobalBar;
|
||||||
|
}
|
||||||
|
+ (void)setFoo:(int)foo;
|
||||||
|
+ (void)setSprite:(int)sprite;
|
||||||
|
- (void)setFoo:(int)foo;
|
||||||
|
- (void)setSprite:(int)sprite;
|
||||||
|
@end
|
||||||
|
|
||||||
|
int spree = 23;
|
||||||
|
int UseGlobalBar;
|
||||||
|
|
||||||
|
@implementation Sprite
|
||||||
|
+ (void)setFoo:(int)foo {
|
||||||
|
sprite = foo; // expected-error {{use of undeclared identifier 'sprite'}}
|
||||||
|
spree = foo;
|
||||||
|
Xsprite = foo; // expected-error {{use of undeclared identifier 'Xsprite'}}
|
||||||
|
UseGlobalBar = 10;
|
||||||
|
}
|
||||||
|
+ (void)setSprite:(int)sprite {
|
||||||
|
int spree;
|
||||||
|
sprite = 15;
|
||||||
|
spree = 17;
|
||||||
|
((Sprite *)self)->sprite = 16; /* NB: This is how one _should_ access */
|
||||||
|
((Sprite *)self)->spree = 18; /* ivars from within class methods! */
|
||||||
|
}
|
||||||
|
- (void)setFoo:(int)foo {
|
||||||
|
sprite = foo;
|
||||||
|
spree = foo;
|
||||||
|
}
|
||||||
|
- (void)setSprite:(int)sprite {
|
||||||
|
int spree;
|
||||||
|
sprite = 15; // expected-warning {{local declaration of 'sprite' hides instance variable}}
|
||||||
|
self->sprite = 16;
|
||||||
|
spree = 17; // expected-warning {{local declaration of 'spree' hides instance variable}}
|
||||||
|
self->spree = 18;
|
||||||
|
}
|
||||||
|
@end
|
Loading…
Reference in New Issue