123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- /**
- * Tencent is pleased to support the open source community by making MLeaksFinder available.
- *
- * Copyright (C) 2017 THL A29 Limited, a Tencent company. All rights reserved.
- *
- * Licensed under the BSD 3-Clause License (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
- *
- * https://opensource.org/licenses/BSD-3-Clause
- *
- * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
- */
- #import "NSObject+MemoryLeak.h"
- #import "MLeakedObjectProxy.h"
- #import "MLeaksFinder.h"
- #import <objc/runtime.h>
- #import <UIKit/UIKit.h>
- #if _INTERNAL_MLF_RC_ENABLED
- #import <FBRetainCycleDetector/FBRetainCycleDetector.h>
- #endif
- static const void *const kViewStackKey = &kViewStackKey;
- static const void *const kParentPtrsKey = &kParentPtrsKey;
- const void *const kLatestSenderKey = &kLatestSenderKey;
- @implementation NSObject (MemoryLeak)
- - (BOOL)willDealloc {
- NSString *className = NSStringFromClass([self class]);
- if ([[NSObject classNamesWhitelist] containsObject:className])
- return NO;
-
- NSNumber *senderPtr = objc_getAssociatedObject([UIApplication sharedApplication], kLatestSenderKey);
- if ([senderPtr isEqualToNumber:@((uintptr_t)self)])
- return NO;
-
- __weak id weakSelf = self;
- dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
- __strong id strongSelf = weakSelf;
- [strongSelf assertNotDealloc];
- });
-
- return YES;
- }
- - (void)assertNotDealloc {
- if ([MLeakedObjectProxy isAnyObjectLeakedAtPtrs:[self parentPtrs]]) {
- return;
- }
- [MLeakedObjectProxy addLeakedObject:self];
-
- NSString *className = NSStringFromClass([self class]);
- NSLog(@"Possibly Memory Leak.\nIn case that %@ should not be dealloced, override -willDealloc in %@ by returning NO.\nView-ViewController stack: %@", className, className, [self viewStack]);
- }
- - (void)willReleaseObject:(id)object relationship:(NSString *)relationship {
- if ([relationship hasPrefix:@"self"]) {
- relationship = [relationship stringByReplacingCharactersInRange:NSMakeRange(0, 4) withString:@""];
- }
- NSString *className = NSStringFromClass([object class]);
- className = [NSString stringWithFormat:@"%@(%@), ", relationship, className];
-
- [object setViewStack:[[self viewStack] arrayByAddingObject:className]];
- [object setParentPtrs:[[self parentPtrs] setByAddingObject:@((uintptr_t)object)]];
- [object willDealloc];
- }
- - (void)willReleaseChild:(id)child {
- if (!child) {
- return;
- }
-
- [self willReleaseChildren:@[ child ]];
- }
- - (void)willReleaseChildren:(NSArray *)children {
- NSArray *viewStack = [self viewStack];
- NSSet *parentPtrs = [self parentPtrs];
- for (id child in children) {
- NSString *className = NSStringFromClass([child class]);
- [child setViewStack:[viewStack arrayByAddingObject:className]];
- [child setParentPtrs:[parentPtrs setByAddingObject:@((uintptr_t)child)]];
- [child willDealloc];
- }
- }
- - (NSArray *)viewStack {
- NSArray *viewStack = objc_getAssociatedObject(self, kViewStackKey);
- if (viewStack) {
- return viewStack;
- }
-
- NSString *className = NSStringFromClass([self class]);
- return @[ className ];
- }
- - (void)setViewStack:(NSArray *)viewStack {
- objc_setAssociatedObject(self, kViewStackKey, viewStack, OBJC_ASSOCIATION_RETAIN);
- }
- - (NSSet *)parentPtrs {
- NSSet *parentPtrs = objc_getAssociatedObject(self, kParentPtrsKey);
- if (!parentPtrs) {
- parentPtrs = [[NSSet alloc] initWithObjects:@((uintptr_t)self), nil];
- }
- return parentPtrs;
- }
- - (void)setParentPtrs:(NSSet *)parentPtrs {
- objc_setAssociatedObject(self, kParentPtrsKey, parentPtrs, OBJC_ASSOCIATION_RETAIN);
- }
- + (NSMutableSet *)classNamesWhitelist {
- static NSMutableSet *whitelist = nil;
- static dispatch_once_t onceToken;
- dispatch_once(&onceToken, ^{
- whitelist = [NSMutableSet setWithObjects:
- @"UIFieldEditor", // UIAlertControllerTextField
- @"UINavigationBar",
- @"_UIAlertControllerActionView",
- @"_UIVisualEffectBackdropView",
- nil];
-
- // System's bug since iOS 10 and not fixed yet up to this ci.
- NSString *systemVersion = [UIDevice currentDevice].systemVersion;
- if ([systemVersion compare:@"10.0" options:NSNumericSearch] != NSOrderedAscending) {
- [whitelist addObject:@"UISwitch"];
- }
- });
- return whitelist;
- }
- + (void)addClassNamesToWhitelist:(NSArray *)classNames {
- [[self classNamesWhitelist] addObjectsFromArray:classNames];
- }
- + (void)swizzleSEL:(SEL)originalSEL withSEL:(SEL)swizzledSEL {
- #if _INTERNAL_MLF_ENABLED
-
- #if _INTERNAL_MLF_RC_ENABLED
- // Just find a place to set up FBRetainCycleDetector.
- static dispatch_once_t onceToken;
- dispatch_once(&onceToken, ^{
- dispatch_async(dispatch_get_main_queue(), ^{
- [FBAssociationManager hook];
- });
- });
- #endif
-
- Class class = [self class];
-
- Method originalMethod = class_getInstanceMethod(class, originalSEL);
- Method swizzledMethod = class_getInstanceMethod(class, swizzledSEL);
-
- BOOL didAddMethod =
- class_addMethod(class,
- originalSEL,
- method_getImplementation(swizzledMethod),
- method_getTypeEncoding(swizzledMethod));
-
- if (didAddMethod) {
- class_replaceMethod(class,
- swizzledSEL,
- method_getImplementation(originalMethod),
- method_getTypeEncoding(originalMethod));
- } else {
- method_exchangeImplementations(originalMethod, swizzledMethod);
- }
- #endif
- }
- @end
|