llvm-project/clang/test/SemaObjC/typo-correction.m
Volodymyr Sapsai 4c33079dc3 [Sema] Correct typos in LHS, RHS before building a binop expression.
Specifically, typo correction should be done before dispatching between
different kinds of binary operations like pseudo-object assignment,
overloaded binary operation, etc.

Without this change we hit an assertion

    Assertion failed: (!LHSExpr->hasPlaceholderType(BuiltinType::PseudoObject)), function CheckAssignmentOperands

when in Objective-C we reference a property without `self` and there are
2 equally good typo correction candidates: ivar and a class name. In
this case LHS expression in `BuildBinOp` is

    CXXDependentScopeMemberExpr
    `-TypoExpr

and instead of handling Obj-C property assignment as pseudo-object
assignment, we call `CreateBuiltinBinOp` which corrects typo to

    ObjCPropertyRefExpr '<pseudo-object type>'

but cannot handle pseudo-objects and asserts about it (indirectly,
through `CheckAssignmentOperands`).

rdar://problem/33102722

Reviewers: rsmith, ahatanak, majnemer

Reviewed By: ahatanak

Subscribers: cfe-commits

Differential Revision: https://reviews.llvm.org/D37322

llvm-svn: 313323
2017-09-15 00:08:37 +00:00

74 lines
1.9 KiB
Objective-C

// RUN: %clang_cc1 %s -verify -fsyntax-only -fobjc-runtime=ios
@protocol P
-(id)description;
@end
@interface B<P>
@property int x;
@end
@interface S : B {
id _someivar; // expected-note {{here}}
}
@end
// Spell-checking 'undefined' is ok.
undefined var; // expected-error {{unknown type name}}
typedef int super1;
@implementation S
-(void)foo:(id)p1 other:(id)p2 {
// Spell-checking 'super' is not ok.
super.x = 0;
self.x = 0;
}
-(void)test {
[self foo:[super description] other:someivar]; // expected-error {{use of undeclared identifier 'someivar'; did you mean '_someivar'?}}
}
@end
__attribute__ (( __objc_root_class__ ))
@interface I {
id _interface; // expected-note {{'_interface' declared here}}
}
-(void)method;
@end
@interface I () {
id _extension; // expected-note {{'_extension' declared here}}
}
@end
@implementation I {
id _implementation; // expected-note {{'_implementation' declared here}}
}
-(void)method {
(void)self->implementation; // expected-error {{'I' does not have a member named 'implementation'; did you mean '_implementation'?}}
(void)self->interface; // expected-error {{'I' does not have a member named 'interface'; did you mean '_interface'?}}
(void)self->extension; // expected-error {{'I' does not have a member named 'extension'; did you mean '_extension'?}}
}
@end
// rdar://problem/33102722
// Typo correction for a property when it has as correction candidates
// synthesized ivar and a class name, both at the same edit distance.
@class TypoCandidate;
__attribute__ (( __objc_root_class__ ))
@interface PropertyType
@property int x;
@end
__attribute__ (( __objc_root_class__ ))
@interface InterfaceC
@property(assign) PropertyType *typoCandidate; // expected-note {{'_typoCandidate' declared here}}
@end
@implementation InterfaceC
-(void)method {
typoCandidate.x = 0; // expected-error {{use of undeclared identifier 'typoCandidate'; did you mean '_typoCandidate'?}}
}
@end