Home | History | Annotate | Download | only in SemaObjC
      1 // RUN: %clang_cc1 -fsyntax-only -Wno-everything -Wobjc-literal-compare "-Dnil=((id)0)" -verify %s
      2 // RUN: %clang_cc1 -fsyntax-only -Wno-everything -Wobjc-literal-compare "-Dnil=(id)0" -verify %s
      3 // RUN: %clang_cc1 -fsyntax-only -Wno-everything -Wobjc-literal-compare "-Dnil=0" -verify %s
      4 
      5 // RUN: %clang_cc1 -fsyntax-only -Wno-everything -Wobjc-literal-compare -fobjc-arc "-Dnil=((id)0)" -verify %s
      6 // RUN: %clang_cc1 -fsyntax-only -Wno-everything -Wobjc-literal-compare -fobjc-arc "-Dnil=(id)0" -verify %s
      7 // RUN: %clang_cc1 -fsyntax-only -Wno-everything -Wobjc-literal-compare -fobjc-arc "-Dnil=0" -verify %s
      8 
      9 // (test the warning flag as well)
     10 
     11 typedef signed char BOOL;
     12 
     13 @interface BaseObject
     14 + (instancetype)new;
     15 @end
     16 
     17 @interface NSObject : BaseObject
     18 - (BOOL)isEqual:(id)other;
     19 @end
     20 
     21 @interface NSNumber : NSObject
     22 + (NSNumber *)numberWithInt:(int)value;
     23 + (NSNumber *)numberWithDouble:(double)value;
     24 + (NSNumber *)numberWithBool:(BOOL)value;
     25 @end
     26 
     27 @interface NSArray : NSObject
     28 + (id)arrayWithObjects:(const id [])objects count:(unsigned long)cnt;
     29 @end
     30 
     31 @interface NSDictionary : NSObject
     32 + (id)dictionaryWithObjects:(const id [])objects forKeys:(const id [])keys count:(unsigned long)cnt;
     33 @end
     34 
     35 @interface NSString : NSObject
     36 @end
     37 
     38 void testComparisonsWithFixits(id obj) {
     39   if (obj == @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     40   if (obj != @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     41   if (@"" == obj) return; // expected-warning{{direct comparison of a string literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     42   if (@"" == @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     43 
     44   if (@[] == obj) return; // expected-warning{{direct comparison of an array literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     45   if (@{} == obj) return; // expected-warning{{direct comparison of a dictionary literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     46   if (@12 == obj) return; // expected-warning{{direct comparison of a numeric literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     47   if (@1.0 == obj) return; // expected-warning{{direct comparison of a numeric literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     48   if (@__objc_yes == obj) return; // expected-warning{{direct comparison of a numeric literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     49   if (@(1+1) == obj) return; // expected-warning{{direct comparison of a boxed expression has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     50 }
     51 
     52 
     53 @interface BadEqualReturnString : NSString
     54 - (void)isEqual:(id)other;
     55 @end
     56 
     57 @interface BadEqualArgString : NSString
     58 - (BOOL)isEqual:(int)other;
     59 @end
     60 
     61 
     62 void testComparisonsWithoutFixits() {
     63   if ([BaseObject new] == @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}}
     64 
     65   if ([BadEqualReturnString new] == @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}}
     66   if ([BadEqualArgString new] == @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}}
     67 
     68   if (@"" < @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}}
     69   if (@"" > @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}}
     70   if (@"" <= @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}}
     71   if (@"" >= @"") return; // expected-warning{{direct comparison of a string literal has undefined behavior}}
     72 }
     73 
     74 
     75 #pragma clang diagnostic push
     76 #pragma clang diagnostic ignored "-Wobjc-string-compare"
     77 
     78 void testWarningFlags(id obj) {
     79   if (obj == @"") return; // no-warning
     80   if (@"" == obj) return; // no-warning
     81 
     82   if (obj == @1) return; // expected-warning{{direct comparison of a numeric literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     83   if (@1 == obj) return; // expected-warning{{direct comparison of a numeric literal has undefined behavior}} expected-note{{use 'isEqual:' instead}}
     84 }
     85 
     86 #pragma clang diagnostic pop
     87 
     88 
     89 void testNilComparison() {
     90   // Don't warn when comparing to nil in a macro.
     91 #define RETURN_IF_NIL(x) if (x == nil || nil == x) return
     92   RETURN_IF_NIL(@"");
     93   RETURN_IF_NIL(@1);
     94   RETURN_IF_NIL(@1.0);
     95   RETURN_IF_NIL(@[]);
     96   RETURN_IF_NIL(@{});
     97   RETURN_IF_NIL(@__objc_yes);
     98   RETURN_IF_NIL(@(1+1));
     99 }
    100 
    101 void PR15257(Class c) {
    102   return c == @""; // expected-warning{{direct comparison of a string literal has undefined behavior}}
    103 }
    104