You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

232 lines
5.3 KiB

/*
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
#include <Foundation/NSObject.h>
@class SelfInBlockTest;
@interface SelfInBlockTestUser
- (void)use_self_in_block_test:(SelfInBlockTest*)test;
- (void)use_self_in_block_test_nullable:(int)x
and:(_Nullable SelfInBlockTest*)test;
@end
@interface A
@property(nonatomic, strong) id image;
@end
@interface SelfInBlockTest : NSObject
@property(nonatomic, weak) SelfInBlockTestUser* user;
- (void)foo;
- (void)bar;
@end
void m(SelfInBlockTest* obj) {}
void m2(_Nullable SelfInBlockTest* obj) {}
@implementation SelfInBlockTest {
int x;
}
- (void)foo {
}
- (void)bar {
}
- (void)mixSelfWeakSelf_bad {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
__strong __typeof(weakSelf) strongSelf = weakSelf;
if (strongSelf) {
[strongSelf foo];
int x = self->x; // bug here
[self foo];
}
return 0;
};
}
- (void)strongSelfNoCheckNotWeakSelf_good {
__typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
__strong __typeof(weakSelf) strongSelf = weakSelf;
return strongSelf->x;
};
}
- (void)strongSelfNoCheck_good {
__weak __typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
__strong __typeof(weakSelf) strongSelf = weakSelf;
[strongSelf foo];
return 0;
};
}
// very unlikely pattern, but still complies with invariant:
// any use of strongSelf is bad unless checked for null beforehand
- (void)strongSelfNoCheck2_good {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
__strong __typeof(weakSelf) strongSelf = weakSelf;
__strong __typeof(weakSelf) strongSelf2 = strongSelf;
return 0;
};
}
- (void)strongSelfCheckOnce_bad {
__weak __typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
__strong __typeof(weakSelf) strongSelf = weakSelf;
if (strongSelf) {
[strongSelf foo];
int x = strongSelf->x;
} else {
strongSelf->x; // bug here
[strongSelf foo]; // no bug here
m(strongSelf); // no bug here because of dedup
}
return 0;
};
}
- (void)strongSelfCheck2_bad {
__weak __typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
__strong __typeof(weakSelf) strongSelf = weakSelf;
if (strongSelf) {
[strongSelf foo];
int x = strongSelf->x;
} else {
m(strongSelf); // bug here
int x = strongSelf->x; // no bug here because of dedup
}
return 0;
};
}
- (void)strongSelfCheck6_good {
__weak __typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
__strong __typeof(weakSelf) strongSelf = weakSelf;
m2(strongSelf); // no bug here because of _Nullable annotation
return 0;
};
}
- (void)strongSelfCheck2_good {
__weak __typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
__strong __typeof(weakSelf) strongSelf = weakSelf;
if (!strongSelf) {
return 0;
} else {
[strongSelf foo];
}
return 0;
};
}
- (void)strongSelfCheck3_bad {
__weak __typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
__strong __typeof(weakSelf) strongSelf = weakSelf;
int y = strongSelf->x;
if (strongSelf == nil) {
return 0;
}
return strongSelf->x;
};
}
- (void)strongSelfCheck4_bad {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
__strong __typeof(weakSelf) strongSelf = weakSelf;
[strongSelf.user use_self_in_block_test:strongSelf]; // bug here
return 0;
};
}
- (void)strongSelfCheck5_good {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
__strong __typeof(weakSelf) strongSelf = weakSelf;
[strongSelf.user
use_self_in_block_test_nullable:1
and:strongSelf]; // no bug here because of
// _Nullable annotation
return 0;
};
}
- (void)strongSelfNotCheck5_good:(A*)a {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
__strong __typeof(weakSelf) strongSelf = weakSelf;
if (strongSelf && a.image) {
int x = strongSelf->x; // no bug here
}
return 0;
};
}
- (void)strongSelfCheck7_good {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
__strong __typeof(weakSelf) strongSelf = weakSelf;
return strongSelf ? strongSelf->x : 0; // no bug here
};
}
- (void)weakSelfMultiple_bad {
__weak __typeof(self) weakSelf = self;
int (^my_block)(BOOL) = ^(BOOL isTapped) {
[weakSelf foo];
[weakSelf bar];
return 0;
};
}
- (void)capturedStrongSelf_bad {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
__strong typeof(self) strongSelf = weakSelf;
if (strongSelf) {
int (^my_block)() = ^() {
int x = strongSelf->x; // bug here
return 0;
};
int x = strongSelf->x;
}
return 0;
};
}
- (void)mixSelfWeakSelf_good:(NSArray*)resources {
__weak __typeof(self) weakSelf = self;
int (^my_block)() = ^() {
[self foo]; // no bug here
int (^my_block)() = ^() {
[weakSelf foo];
return 0;
};
return 0;
};
}
@end