1// RUN: %clang_cc1 -fsyntax-only -fobjc-runtime-has-weak -fobjc-arc -fblocks -Wno-objc-root-class -std=c++11 -Warc-repeated-use-of-weak -verify %s 2 3@interface Test { 4@public 5 Test *ivar; 6 __weak id weakIvar; 7} 8@property(weak) Test *weakProp; 9@property(strong) Test *strongProp; 10 11- (__weak id)implicitProp; 12 13+ (__weak id)weakProp; 14@end 15 16extern void use(id); 17extern id get(); 18extern bool condition(); 19#define nil ((id)0) 20 21void sanity(Test *a) { 22 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times in this function but may be unpredictably set to nil; assign to a strong variable to keep the object alive}} 23 use(a.weakProp); // expected-note{{also accessed here}} 24 25 use(a.strongProp); 26 use(a.strongProp); // no-warning 27 28 use(a.weakProp); // expected-note{{also accessed here}} 29} 30 31void singleUse(Test *a) { 32 use(a.weakProp); // no-warning 33 use(a.strongProp); // no-warning 34} 35 36void assignsOnly(Test *a) { 37 a.weakProp = get(); // no-warning 38 39 id next = get(); 40 if (next) 41 a.weakProp = next; // no-warning 42 43 a->weakIvar = get(); // no-warning 44 next = get(); 45 if (next) 46 a->weakIvar = next; // no-warning 47 48 extern __weak id x; 49 x = get(); // no-warning 50 next = get(); 51 if (next) 52 x = next; // no-warning 53} 54 55void assignThenRead(Test *a) { 56 a.weakProp = get(); // expected-note{{also accessed here}} 57 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 58} 59 60void twoVariables(Test *a, Test *b) { 61 use(a.weakProp); // no-warning 62 use(b.weakProp); // no-warning 63} 64 65void doubleLevelAccess(Test *a) { 66 use(a.strongProp.weakProp); // expected-warning{{weak property 'weakProp' may be accessed multiple times in this function and may be unpredictably set to nil; assign to a strong variable to keep the object alive}} 67 use(a.strongProp.weakProp); // expected-note{{also accessed here}} 68} 69 70void doubleLevelAccessIvar(Test *a) { 71 use(a.strongProp.weakProp); // expected-warning{{weak property 'weakProp' may be accessed multiple times}} 72 use(a.strongProp.weakProp); // expected-note{{also accessed here}} 73} 74 75void implicitProperties(Test *a) { 76 use(a.implicitProp); // expected-warning{{weak implicit property 'implicitProp' is accessed multiple times}} 77 use(a.implicitProp); // expected-note{{also accessed here}} 78} 79 80void classProperties() { 81 use(Test.weakProp); // expected-warning{{weak implicit property 'weakProp' is accessed multiple times}} 82 use(Test.weakProp); // expected-note{{also accessed here}} 83} 84 85void classPropertiesAreDifferent(Test *a) { 86 use(Test.weakProp); // no-warning 87 use(a.weakProp); // no-warning 88 use(a.strongProp.weakProp); // no-warning 89} 90 91void ivars(Test *a) { 92 use(a->weakIvar); // expected-warning{{weak instance variable 'weakIvar' is accessed multiple times}} 93 use(a->weakIvar); // expected-note{{also accessed here}} 94} 95 96void globals() { 97 extern __weak id a; 98 use(a); // expected-warning{{weak variable 'a' is accessed multiple times}} 99 use(a); // expected-note{{also accessed here}} 100} 101 102void messageGetter(Test *a) { 103 use([a weakProp]); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 104 use([a weakProp]); // expected-note{{also accessed here}} 105} 106 107void messageSetter(Test *a) { 108 [a setWeakProp:get()]; // no-warning 109 [a setWeakProp:get()]; // no-warning 110} 111 112void messageSetterAndGetter(Test *a) { 113 [a setWeakProp:get()]; // expected-note{{also accessed here}} 114 use([a weakProp]); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 115} 116 117void mixDotAndMessageSend(Test *a, Test *b) { 118 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 119 use([a weakProp]); // expected-note{{also accessed here}} 120 121 use([b weakProp]); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 122 use(b.weakProp); // expected-note{{also accessed here}} 123} 124 125 126void assignToStrongWrongInit(Test *a) { 127 id val = a.weakProp; // expected-note{{also accessed here}} 128 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 129} 130 131void assignToStrongWrong(Test *a) { 132 id val; 133 val = a.weakProp; // expected-note{{also accessed here}} 134 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 135} 136 137void assignToIvarWrong(Test *a) { 138 a->weakIvar = get(); // expected-note{{also accessed here}} 139 use(a->weakIvar); // expected-warning{{weak instance variable 'weakIvar' is accessed multiple times}} 140} 141 142void assignToGlobalWrong() { 143 extern __weak id a; 144 a = get(); // expected-note{{also accessed here}} 145 use(a); // expected-warning{{weak variable 'a' is accessed multiple times}} 146} 147 148void assignToStrongOK(Test *a) { 149 if (condition()) { 150 id val = a.weakProp; // no-warning 151 (void)val; 152 } else { 153 id val; 154 val = a.weakProp; // no-warning 155 (void)val; 156 } 157} 158 159void assignToStrongConditional(Test *a) { 160 id val = (condition() ? a.weakProp : a.weakProp); // no-warning 161 id val2 = a.implicitProp ?: a.implicitProp; // no-warning 162} 163 164void testBlock(Test *a) { 165 use(a.weakProp); // no-warning 166 167 use(^{ 168 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times in this block}} 169 use(a.weakProp); // expected-note{{also accessed here}} 170 }); 171} 172 173void assignToStrongWithCasts(Test *a) { 174 if (condition()) { 175 Test *val = (Test *)a.weakProp; // no-warning 176 (void)val; 177 } else { 178 id val; 179 val = (Test *)a.weakProp; // no-warning 180 (void)val; 181 } 182} 183 184void assignToStrongWithMessages(Test *a) { 185 if (condition()) { 186 id val = [a weakProp]; // no-warning 187 (void)val; 188 } else { 189 id val; 190 val = [a weakProp]; // no-warning 191 (void)val; 192 } 193} 194 195 196void assignAfterRead(Test *a) { 197 // Special exception for a single read before any writes. 198 if (!a.weakProp) // no-warning 199 a.weakProp = get(); // no-warning 200} 201 202void readOnceWriteMany(Test *a) { 203 if (!a.weakProp) { // no-warning 204 a.weakProp = get(); // no-warning 205 a.weakProp = get(); // no-warning 206 } 207} 208 209void readOnceAfterWrite(Test *a) { 210 a.weakProp = get(); // expected-note{{also accessed here}} 211 if (!a.weakProp) { // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 212 a.weakProp = get(); // expected-note{{also accessed here}} 213 } 214} 215 216void readOnceWriteManyLoops(Test *a, Test *b, Test *c, Test *d, Test *e) { 217 while (condition()) { 218 if (!a.weakProp) { // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 219 a.weakProp = get(); // expected-note{{also accessed here}} 220 a.weakProp = get(); // expected-note{{also accessed here}} 221 } 222 } 223 224 do { 225 if (!b.weakProp) { // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 226 b.weakProp = get(); // expected-note{{also accessed here}} 227 b.weakProp = get(); // expected-note{{also accessed here}} 228 } 229 } while (condition()); 230 231 for (id x = get(); x; x = get()) { 232 if (!c.weakProp) { // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 233 c.weakProp = get(); // expected-note{{also accessed here}} 234 c.weakProp = get(); // expected-note{{also accessed here}} 235 } 236 } 237 238 for (id x in get()) { 239 if (!d.weakProp) { // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 240 d.weakProp = get(); // expected-note{{also accessed here}} 241 d.weakProp = get(); // expected-note{{also accessed here}} 242 } 243 } 244 245 int array[] = { 1, 2, 3 }; 246 for (int i : array) { 247 if (!e.weakProp) { // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 248 e.weakProp = get(); // expected-note{{also accessed here}} 249 e.weakProp = get(); // expected-note{{also accessed here}} 250 } 251 } 252} 253 254void readOnlyLoop(Test *a) { 255 while (condition()) { 256 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 257 } 258} 259 260void readInIterationLoop() { 261 for (Test *a in get()) 262 use(a.weakProp); // no-warning 263} 264 265void readDoubleLevelAccessInLoop() { 266 for (Test *a in get()) { 267 use(a.strongProp.weakProp); // no-warning 268 } 269} 270 271void readParameterInLoop(Test *a) { 272 for (id unused in get()) { 273 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times in this function}} 274 (void)unused; 275 } 276} 277 278void readGlobalInLoop() { 279 static __weak id a; 280 for (id unused in get()) { 281 use(a); // expected-warning{{weak variable 'a' is accessed multiple times in this function}} 282 (void)unused; 283 } 284} 285 286void doWhileLoop(Test *a) { 287 do { 288 use(a.weakProp); // no-warning 289 } while(0); 290} 291 292 293@interface Test (Methods) 294@end 295 296@implementation Test (Methods) 297- (void)sanity { 298 use(self.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times in this method but may be unpredictably set to nil; assign to a strong variable to keep the object alive}} 299 use(self.weakProp); // expected-note{{also accessed here}} 300} 301 302- (void)ivars { 303 use(weakIvar); // expected-warning{{weak instance variable 'weakIvar' is accessed multiple times in this method but may be unpredictably set to nil; assign to a strong variable to keep the object alive}} 304 use(weakIvar); // expected-note{{also accessed here}} 305} 306 307- (void)doubleLevelAccessForSelf { 308 use(self.strongProp.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 309 use(self.strongProp.weakProp); // expected-note{{also accessed here}} 310 311 use(self->ivar.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 312 use(self->ivar.weakProp); // expected-note{{also accessed here}} 313 314 use(self->ivar->weakIvar); // expected-warning{{weak instance variable 'weakIvar' is accessed multiple times}} 315 use(self->ivar->weakIvar); // expected-note{{also accessed here}} 316} 317 318- (void)distinctFromOther:(Test *)other { 319 use(self.strongProp.weakProp); // no-warning 320 use(other.strongProp.weakProp); // no-warning 321 322 use(self->ivar.weakProp); // no-warning 323 use(other->ivar.weakProp); // no-warning 324 325 use(self.strongProp->weakIvar); // no-warning 326 use(other.strongProp->weakIvar); // no-warning 327} 328@end 329 330@interface Base1 331@end 332@interface Sub1 : Base1 333@end 334@interface Sub1(cat) 335-(id)prop; 336@end 337 338void test1(Sub1 *s) { 339 use([s prop]); 340 use([s prop]); 341} 342 343@interface Base1(cat) 344@property (weak) id prop; 345@end 346 347void test2(Sub1 *s) { 348 // This does not warn because the "prop" in "Base1(cat)" was introduced 349 // after the method declaration and we don't find it as overridden. 350 // Always looking for overridden methods after the method declaration is expensive 351 // and it's not clear it is worth it currently. 352 use([s prop]); 353 use([s prop]); 354} 355 356 357class Wrapper { 358 Test *a; 359 360public: 361 void fields() { 362 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times in this function but may be unpredictably set to nil; assign to a strong variable to keep the object alive}} 363 use(a.weakProp); // expected-note{{also accessed here}} 364 } 365 366 void distinctFromOther(Test *b, const Wrapper &w) { 367 use(a.weakProp); // no-warning 368 use(b.weakProp); // no-warning 369 use(w.a.weakProp); // no-warning 370 } 371 372 static void doubleLevelAccessField(const Wrapper &x, const Wrapper &y) { 373 use(x.a.weakProp); // expected-warning{{weak property 'weakProp' may be accessed multiple times}} 374 use(y.a.weakProp); // expected-note{{also accessed here}} 375 } 376}; 377 378 379// ----------------------- 380// False positives 381// ----------------------- 382 383// Most of these would require flow-sensitive analysis to silence correctly. 384 385void assignNil(Test *a) { 386 if (condition()) 387 a.weakProp = nil; // expected-note{{also accessed here}} 388 389 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 390} 391 392void branch(Test *a) { 393 if (condition()) 394 use(a.weakProp); // expected-warning{{weak property 'weakProp' is accessed multiple times}} 395 else 396 use(a.weakProp); // expected-note{{also accessed here}} 397} 398 399void doubleLevelAccess(Test *a, Test *b) { 400 use(a.strongProp.weakProp); // expected-warning{{weak property 'weakProp' may be accessed multiple times}} 401 use(b.strongProp.weakProp); // expected-note{{also accessed here}} 402 403 use(a.weakProp.weakProp); // no-warning 404} 405 406void doubleLevelAccessIvar(Test *a, Test *b) { 407 use(a->ivar.weakProp); // expected-warning{{weak property 'weakProp' may be accessed multiple times}} 408 use(b->ivar.weakProp); // expected-note{{also accessed here}} 409 410 use(a.strongProp.weakProp); // no-warning 411} 412 413// rdar://13942025 414@interface X 415@end 416 417@implementation X 418- (int) warningAboutWeakVariableInsideTypeof { 419 __typeof__(self) __weak weakSelf = self; 420 ^(){ 421 __typeof__(weakSelf) blockSelf = weakSelf; 422 use(blockSelf); 423 }(); 424 return sizeof(weakSelf); 425} 426@end 427 428