mirror of
https://github.com/status-im/react-native.git
synced 2025-01-17 21:11:45 +00:00
3d6ffcf903
Summary: @public When the profile is initialized, it automatically hooks into every method of the bridge modules, that was causing `+initialize` to be called twice. Also add a popup to notify the user that the profile has been created. Test Plan: Run the UIExplorer, start the profiler, try to rage shake to open the dev menu again. It should now work, and show an alertview with some information once the profile is stopped.
332 lines
9.4 KiB
Objective-C
332 lines
9.4 KiB
Objective-C
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*/
|
|
|
|
#import "RCTProfile.h"
|
|
|
|
#import <mach/mach.h>
|
|
#import <objc/message.h>
|
|
#import <objc/runtime.h>
|
|
|
|
#import <UIKit/UIKit.h>
|
|
|
|
#import "RCTAssert.h"
|
|
#import "RCTBridge.h"
|
|
#import "RCTDefines.h"
|
|
#import "RCTUtils.h"
|
|
|
|
NSString *const RCTProfileDidStartProfiling = @"RCTProfileDidStartProfiling";
|
|
NSString *const RCTProfileDidEndProfiling = @"RCTProfileDidEndProfiling";
|
|
|
|
#if RCT_DEV
|
|
|
|
#pragma mark - Prototypes
|
|
|
|
NSNumber *RCTProfileTimestamp(NSTimeInterval);
|
|
NSString *RCTProfileMemory(vm_size_t);
|
|
NSDictionary *RCTProfileGetMemoryUsage(void);
|
|
|
|
#pragma mark - Constants
|
|
|
|
NSString const *RCTProfileTraceEvents = @"traceEvents";
|
|
NSString const *RCTProfileSamples = @"samples";
|
|
NSString *const RCTProfilePrefix = @"rct_profile_";
|
|
|
|
#pragma mark - Variables
|
|
|
|
NSDictionary *RCTProfileInfo;
|
|
NSUInteger RCTProfileEventID = 0;
|
|
NSMutableDictionary *RCTProfileOngoingEvents;
|
|
NSTimeInterval RCTProfileStartTime;
|
|
NSRecursiveLock *_RCTProfileLock;
|
|
|
|
#pragma mark - Macros
|
|
|
|
#define RCTProfileAddEvent(type, props...) \
|
|
[RCTProfileInfo[type] addObject:@{ \
|
|
@"pid": @([[NSProcessInfo processInfo] processIdentifier]), \
|
|
@"tid": RCTCurrentThreadName(), \
|
|
props \
|
|
}];
|
|
|
|
#define CHECK(...) \
|
|
if (!RCTProfileIsProfiling()) { \
|
|
return __VA_ARGS__; \
|
|
}
|
|
|
|
#define RCTProfileLock(...) \
|
|
[_RCTProfileLock lock]; \
|
|
__VA_ARGS__ \
|
|
[_RCTProfileLock unlock]
|
|
|
|
#pragma mark - Private Helpers
|
|
|
|
NSNumber *RCTProfileTimestamp(NSTimeInterval timestamp)
|
|
{
|
|
return @((timestamp - RCTProfileStartTime) * 1e6);
|
|
}
|
|
|
|
NSString *RCTProfileMemory(vm_size_t memory)
|
|
{
|
|
double mem = ((double)memory) / 1024 / 1024;
|
|
return [NSString stringWithFormat:@"%.2lfmb", mem];
|
|
}
|
|
|
|
NSDictionary *RCTProfileGetMemoryUsage(void)
|
|
{
|
|
struct task_basic_info info;
|
|
mach_msg_type_number_t size = sizeof(info);
|
|
kern_return_t kerr = task_info(mach_task_self(),
|
|
TASK_BASIC_INFO,
|
|
(task_info_t)&info,
|
|
&size);
|
|
if( kerr == KERN_SUCCESS ) {
|
|
return @{
|
|
@"suspend_count": @(info.suspend_count),
|
|
@"virtual_size": RCTProfileMemory(info.virtual_size),
|
|
@"resident_size": RCTProfileMemory(info.resident_size),
|
|
};
|
|
} else {
|
|
return @{};
|
|
}
|
|
}
|
|
|
|
#pragma mark - Module hooks
|
|
|
|
@interface RCTBridge (Private)
|
|
|
|
- (void)dispatchBlock:(dispatch_block_t)block forModule:(id<RCTBridgeModule>)module;
|
|
|
|
@end
|
|
|
|
static const char *RCTProfileProxyClassName(Class);
|
|
static const char *RCTProfileProxyClassName(Class class)
|
|
{
|
|
return [RCTProfilePrefix stringByAppendingString:NSStringFromClass(class)].UTF8String;
|
|
}
|
|
|
|
static SEL RCTProfileProxySelector(SEL);
|
|
static SEL RCTProfileProxySelector(SEL selector)
|
|
{
|
|
NSString *selectorName = NSStringFromSelector(selector);
|
|
return NSSelectorFromString([RCTProfilePrefix stringByAppendingString:selectorName]);
|
|
}
|
|
|
|
static void RCTProfileForwardInvocation(NSObject *, SEL, NSInvocation *);
|
|
static void RCTProfileForwardInvocation(NSObject *self, __unused SEL cmd, NSInvocation *invocation)
|
|
{
|
|
NSString *name = [NSString stringWithFormat:@"-[%@ %@]", NSStringFromClass([self class]), NSStringFromSelector(invocation.selector)];
|
|
SEL newSel = RCTProfileProxySelector(invocation.selector);
|
|
|
|
RCTProfileLock(
|
|
if ([object_getClass(self) instancesRespondToSelector:newSel]) {
|
|
invocation.selector = newSel;
|
|
RCTProfileBeginEvent();
|
|
[invocation invoke];
|
|
RCTProfileEndEvent(name, @"objc_call,modules,auto", nil);
|
|
} else {
|
|
// Use original selector to don't change error message
|
|
[self doesNotRecognizeSelector:invocation.selector];
|
|
}
|
|
);
|
|
}
|
|
|
|
static IMP RCTProfileMsgForward(NSObject *, SEL);
|
|
static IMP RCTProfileMsgForward(NSObject *self, SEL selector)
|
|
{
|
|
IMP imp = _objc_msgForward;
|
|
#if !defined(__arm64__)
|
|
NSMethodSignature *signature = [self methodSignatureForSelector:selector];
|
|
if (signature.methodReturnType[0] == _C_STRUCT_B && signature.methodReturnLength > 8) {
|
|
imp = _objc_msgForward_stret;
|
|
}
|
|
#endif
|
|
return imp;
|
|
}
|
|
|
|
static void RCTProfileHookModules(RCTBridge *);
|
|
static void RCTProfileHookModules(RCTBridge *bridge)
|
|
{
|
|
for (id<RCTBridgeModule> module in bridge.modules.allValues) {
|
|
[bridge dispatchBlock:^{
|
|
Class moduleClass = object_getClass(module);
|
|
Class proxyClass = objc_allocateClassPair(moduleClass, RCTProfileProxyClassName(moduleClass), 0);
|
|
|
|
unsigned int methodCount;
|
|
Method *methods = class_copyMethodList(moduleClass, &methodCount);
|
|
for (NSUInteger i = 0; i < methodCount; i++) {
|
|
Method method = methods[i];
|
|
SEL selector = method_getName(method);
|
|
if ([NSStringFromSelector(selector) hasPrefix:@"rct"] || [NSObject instancesRespondToSelector:selector]) {
|
|
continue;
|
|
}
|
|
IMP originalIMP = method_getImplementation(method);
|
|
const char *returnType = method_getTypeEncoding(method);
|
|
class_addMethod(proxyClass, selector, RCTProfileMsgForward(module, selector), returnType);
|
|
class_addMethod(proxyClass, RCTProfileProxySelector(selector), originalIMP, returnType);
|
|
}
|
|
free(methods);
|
|
|
|
class_replaceMethod(object_getClass(proxyClass), @selector(initialize), imp_implementationWithBlock(^{}), "v@:");
|
|
|
|
for (Class cls in @[proxyClass, object_getClass(proxyClass)]) {
|
|
Method oldImp = class_getInstanceMethod(cls, @selector(class));
|
|
class_replaceMethod(cls, @selector(class), imp_implementationWithBlock(^{ return moduleClass; }), method_getTypeEncoding(oldImp));
|
|
}
|
|
|
|
IMP originalFwd = class_replaceMethod(moduleClass, @selector(forwardInvocation:), (IMP)RCTProfileForwardInvocation, "v@:@");
|
|
if (originalFwd != NULL) {
|
|
class_addMethod(proxyClass, RCTProfileProxySelector(@selector(forwardInvocation:)), originalFwd, "v@:@");
|
|
}
|
|
|
|
objc_registerClassPair(proxyClass);
|
|
object_setClass(module, proxyClass);
|
|
} forModule:module];
|
|
}
|
|
}
|
|
|
|
void RCTProfileUnhookModules(RCTBridge *);
|
|
void RCTProfileUnhookModules(RCTBridge *bridge)
|
|
{
|
|
for (id<RCTBridgeModule> module in bridge.modules.allValues) {
|
|
[bridge dispatchBlock:^{
|
|
RCTProfileLock(
|
|
Class proxyClass = object_getClass(module);
|
|
if (module.class != proxyClass) {
|
|
object_setClass(module, module.class);
|
|
objc_disposeClassPair(proxyClass);
|
|
}
|
|
);
|
|
} forModule:module];
|
|
};
|
|
}
|
|
|
|
|
|
#pragma mark - Public Functions
|
|
|
|
BOOL RCTProfileIsProfiling(void)
|
|
{
|
|
RCTProfileLock(
|
|
BOOL profiling = RCTProfileInfo != nil;
|
|
);
|
|
return profiling;
|
|
}
|
|
|
|
void RCTProfileInit(RCTBridge *bridge)
|
|
{
|
|
RCTProfileHookModules(bridge);
|
|
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
_RCTProfileLock = [[NSRecursiveLock alloc] init];
|
|
});
|
|
RCTProfileLock(
|
|
RCTProfileStartTime = CACurrentMediaTime();
|
|
RCTProfileOngoingEvents = [[NSMutableDictionary alloc] init];
|
|
RCTProfileInfo = @{
|
|
RCTProfileTraceEvents: [[NSMutableArray alloc] init],
|
|
RCTProfileSamples: [[NSMutableArray alloc] init],
|
|
};
|
|
);
|
|
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTProfileDidStartProfiling
|
|
object:nil];
|
|
}
|
|
|
|
NSString *RCTProfileEnd(RCTBridge *bridge)
|
|
{
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTProfileDidEndProfiling
|
|
object:nil];
|
|
|
|
RCTProfileLock(
|
|
NSString *log = RCTJSONStringify(RCTProfileInfo, NULL);
|
|
RCTProfileEventID = 0;
|
|
RCTProfileInfo = nil;
|
|
RCTProfileOngoingEvents = nil;
|
|
);
|
|
|
|
RCTProfileUnhookModules(bridge);
|
|
|
|
return log;
|
|
}
|
|
|
|
NSNumber *_RCTProfileBeginEvent(void)
|
|
{
|
|
CHECK(@0);
|
|
RCTProfileLock(
|
|
NSNumber *eventID = @(++RCTProfileEventID);
|
|
RCTProfileOngoingEvents[eventID] = RCTProfileTimestamp(CACurrentMediaTime());
|
|
);
|
|
return eventID;
|
|
}
|
|
|
|
void _RCTProfileEndEvent(NSNumber *eventID, NSString *name, NSString *categories, id args)
|
|
{
|
|
CHECK();
|
|
RCTProfileLock(
|
|
NSNumber *startTimestamp = RCTProfileOngoingEvents[eventID];
|
|
if (startTimestamp) {
|
|
NSNumber *endTimestamp = RCTProfileTimestamp(CACurrentMediaTime());
|
|
|
|
RCTProfileAddEvent(RCTProfileTraceEvents,
|
|
@"name": name,
|
|
@"cat": categories,
|
|
@"ph": @"X",
|
|
@"ts": startTimestamp,
|
|
@"dur": @(endTimestamp.doubleValue - startTimestamp.doubleValue),
|
|
@"args": args ?: @[],
|
|
);
|
|
[RCTProfileOngoingEvents removeObjectForKey:eventID];
|
|
}
|
|
);
|
|
}
|
|
|
|
void RCTProfileImmediateEvent(NSString *name, NSTimeInterval timestamp, NSString *scope)
|
|
{
|
|
CHECK();
|
|
RCTProfileLock(
|
|
RCTProfileAddEvent(RCTProfileTraceEvents,
|
|
@"name": name,
|
|
@"ts": RCTProfileTimestamp(timestamp),
|
|
@"scope": scope,
|
|
@"ph": @"i",
|
|
@"args": RCTProfileGetMemoryUsage(),
|
|
);
|
|
);
|
|
}
|
|
|
|
NSNumber *_RCTProfileBeginFlowEvent(void)
|
|
{
|
|
static NSUInteger flowID = 0;
|
|
|
|
CHECK(@0);
|
|
RCTProfileAddEvent(RCTProfileTraceEvents,
|
|
@"name": @"flow",
|
|
@"id": @(++flowID),
|
|
@"cat": @"flow",
|
|
@"ph": @"s",
|
|
@"ts": RCTProfileTimestamp(CACurrentMediaTime()),
|
|
);
|
|
|
|
return @(flowID);
|
|
}
|
|
|
|
void _RCTProfileEndFlowEvent(NSNumber *flowID)
|
|
{
|
|
CHECK();
|
|
RCTProfileAddEvent(RCTProfileTraceEvents,
|
|
@"name": @"flow",
|
|
@"id": flowID,
|
|
@"cat": @"flow",
|
|
@"ph": @"f",
|
|
@"ts": RCTProfileTimestamp(CACurrentMediaTime()),
|
|
);
|
|
}
|
|
|
|
#endif
|