2015-04-20 04:55:05 -07:00
/ * *
* Copyright ( c ) 2015 - present , Facebook , Inc .
*
2018-02-16 18:24:55 -08:00
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree .
2015-04-20 04:55:05 -07:00
* /
# import "RCTProfile.h"
2015-10-20 04:11:38 -07:00
# import < dlfcn . h >
2015-04-20 04:55:05 -07:00
# import < mach / mach . h >
2015-06-05 04:23:51 -07:00
# import < objc / message . h >
# import < objc / runtime . h >
2017-09-26 13:55:58 -07:00
# import < stdatomic . h >
2015-04-20 04:55:05 -07:00
# import < UIKit / UIKit . h >
2015-05-25 05:19:53 -07:00
# import "RCTAssert.h"
2015-12-15 05:39:30 -08:00
# import "RCTBridge+Private.h"
2016-12-07 16:27:55 -08:00
# import "RCTBridge.h"
2015-12-14 13:07:24 -08:00
# import "RCTComponentData.h"
2015-04-21 05:26:51 -07:00
# import "RCTDefines.h"
2015-09-11 06:35:25 -07:00
# import "RCTLog.h"
[ReactNative] Move module info from bridge to RCTModuleData
Summary:
@public
The info about bridge modules (such as id, name, queue, methods...) was spread
across arrays & dictionaries on the bridge, move it into a specific class.
It also removes a lot of information that was statically cached, and now have
the same lifecycle of the bridge.
Also moved RCTModuleMethod, RCTFrameUpdate and RCTBatchedBridge into it's own
files, for organization sake.
NOTE: This diff seems huge, but most of it was just moving code :)
Test Plan:
Tested UIExplorer & UIExplorer tests, Catalyst, MAdMan and Groups. Everything
looks fine.
2015-06-24 16:34:56 -07:00
# import "RCTModuleData.h"
2015-12-14 13:07:24 -08:00
# import "RCTUIManager.h"
2017-09-26 13:55:58 -07:00
# import "RCTUIManagerUtils.h"
2016-12-07 16:27:55 -08:00
# import "RCTUtils.h"
2015-04-20 04:55:05 -07:00
2015-06-02 06:15:53 -07:00
NSString * const RCTProfileDidStartProfiling = @ "RCTProfileDidStartProfiling" ;
NSString * const RCTProfileDidEndProfiling = @ "RCTProfileDidEndProfiling" ;
2016-05-13 17:14:59 -07:00
const uint64_t RCTProfileTagAlways = 1 L < < 0 ;
2016-06-13 04:16:19 -07:00
# if RCT_PROFILE
2015-04-20 12:33:52 -01:00
2015-04-20 04:55:05 -07:00
# pragma mark - Constants
2017-07-25 04:40:19 -07:00
static NSString * const kProfileTraceEvents = @ "traceEvents" ;
static NSString * const kProfileSamples = @ "samples" ;
static NSString * const kProfilePrefix = @ "rct_profile_" ;
2015-04-20 04:55:05 -07:00
# pragma mark - Variables
2017-08-01 03:51:33 -07:00
static atomic_bool RCTProfileProfiling = ATOMIC_VAR _INIT ( NO ) ;
2015-11-07 13:46:35 -08:00
2015-09-23 11:59:44 -07:00
static NSDictionary * RCTProfileInfo ;
static NSMutableDictionary * RCTProfileOngoingEvents ;
static NSTimeInterval RCTProfileStartTime ;
static NSUInteger RCTProfileEventID = 0 ;
2015-11-23 09:48:43 -08:00
static CADisplayLink * RCTProfileDisplayLink ;
2016-03-11 06:20:24 -08:00
static __weak RCTBridge * _RCTProfilingBridge ;
static UIWindow * RCTProfileControlsWindow ;
2015-04-20 04:55:05 -07:00
# pragma mark - Macros
# define RCTProfileAddEvent ( type , props . . . ) \
[ RCTProfileInfo [ type ] addObject : @ { \
@ "pid" : @ ( [ [ NSProcessInfo processInfo ] processIdentifier ] ) , \
props \
} ] ;
# define CHECK ( . . . ) \
if ( ! RCTProfileIsProfiling ( ) ) { \
return __VA _ARGS __ ; \
}
2015-09-23 11:59:44 -07:00
# pragma mark - systrace glue code
static RCTProfileCallbacks * callbacks ;
static char * systrace_buffer ;
2017-05-15 03:36:58 -07:00
static systrace_arg _t * newSystraceArgsFromDictionary ( NSDictionary < NSString * , NSString * > * args )
2015-09-23 11:59:44 -07:00
{
if ( args . count = = 0 ) {
return NULL ;
}
systrace_arg _t * systrace_args = malloc ( sizeof ( systrace_arg _t ) * args . count ) ;
2018-08-08 18:24:47 -07:00
if ( systrace_args ) {
__block size_t i = 0 ;
[ args enumerateKeysAndObjectsUsingBlock : ^ ( NSString * key , NSString * value , __unused BOOL * stop ) {
systrace_args [ i ] . key = [ key UTF8String ] ;
systrace_args [ i ] . key_len = [ key length ] ;
systrace_args [ i ] . value = [ value UTF8String ] ;
systrace_args [ i ] . value_len = [ value length ] ;
i + + ;
} ] ;
}
2015-09-23 11:59:44 -07:00
return systrace_args ;
}
void RCTProfileRegisterCallbacks ( RCTProfileCallbacks * cb )
{
callbacks = cb ;
}
2015-04-22 07:03:55 -07:00
2015-04-20 04:55:05 -07:00
# pragma mark - Private Helpers
2016-03-11 06:20:24 -08:00
static RCTBridge * RCTProfilingBridge ( void )
{
return _RCTProfilingBridge ? : [ RCTBridge currentBridge ] ;
}
2015-08-19 23:36:11 -07:00
static NSNumber * RCTProfileTimestamp ( NSTimeInterval timestamp )
2015-04-20 04:55:05 -07:00
{
return @ ( ( timestamp - RCTProfileStartTime ) * 1 e6 ) ;
}
2015-08-19 23:36:11 -07:00
static NSString * RCTProfileMemory ( vm_size _t memory )
2015-04-20 04:55:05 -07:00
{
double mem = ( ( double ) memory ) / 1024 / 1024 ;
return [ NSString stringWithFormat : @ "%.2lfmb" , mem ] ;
}
2015-08-21 10:15:04 -07:00
static NSDictionary * RCTProfileGetMemoryUsage ( void )
2015-04-20 04:55:05 -07:00
{
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 ) ;
2017-05-12 17:57:12 -07:00
if ( kerr = = KERN_SUCCESS ) {
2015-04-20 04:55:05 -07:00
return @ {
@ "suspend_count" : @ ( info . suspend_count ) ,
2015-08-21 10:15:04 -07:00
@ "virtual_size" : RCTProfileMemory ( info . virtual_size ) ,
@ "resident_size" : RCTProfileMemory ( info . resident_size ) ,
2015-04-20 04:55:05 -07:00
} ;
} else {
return @ { } ;
}
}
2015-06-05 04:23:51 -07:00
# pragma mark - Module hooks
static const char * RCTProfileProxyClassName ( Class class )
{
2017-07-25 04:40:19 -07:00
return [ kProfilePrefix stringByAppendingString : NSStringFromClass ( class ) ] . UTF8String ;
2015-06-05 04:23:51 -07:00
}
2015-10-02 04:16:04 -07:00
static dispatch_group _t RCTProfileGetUnhookGroup ( void )
{
static dispatch_group _t unhookGroup ;
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
unhookGroup = dispatch_group _create ( ) ;
} ) ;
return unhookGroup ;
}
2017-03-16 12:24:11 -07:00
// Used by RCTProfileTrampoline assembly file to call libc` malloc
RCT_EXTERN void * RCTProfileMalloc ( size_t size ) ;
void * RCTProfileMalloc ( size_t size )
{
return malloc ( size ) ;
}
// Used by RCTProfileTrampoline assembly file to call libc` free
RCT_EXTERN void RCTProfileFree ( void * buf ) ;
void RCTProfileFree ( void * buf )
{
free ( buf ) ;
}
2015-10-20 04:11:38 -07:00
RCT_EXTERN IMP RCTProfileGetImplementation ( id obj , SEL cmd ) ;
IMP RCTProfileGetImplementation ( id obj , SEL cmd )
2015-06-05 04:23:51 -07:00
{
2015-10-20 04:11:38 -07:00
return class_getMethodImplementation ( [ obj class ] , cmd ) ;
2015-06-05 04:23:51 -07:00
}
2015-10-20 04:11:38 -07:00
/ * *
* For the profiling we have to execute some code before and after every
* function being profiled , the only way of doing that with pure Objective - C is
* by using ` - forwardInvocation : ` , which is slow and could skew the profile
* results .
*
* The alternative in assembly is much simpler , we just need to store all the
* state at the beginning of the function , start the profiler , restore all the
* state , call the actual function we want to profile and stop the profiler .
*
* The implementation can be found in RCTProfileTrampoline - < arch > . s where arch
2015-11-24 06:48:04 -08:00
* is one of : i386 , x86_64 , arm , arm64 .
2015-10-20 04:11:38 -07:00
* /
2015-11-24 06:48:04 -08:00
# if defined ( __i386 __ ) || \
2015-10-27 12:59:27 -07:00
defined ( __x86 _64 __ ) || \
defined ( __arm __ ) || \
defined ( __arm64 __ )
2015-10-20 04:11:38 -07:00
2015-10-27 12:59:27 -07:00
RCT_EXTERN void RCTProfileTrampoline ( void ) ;
# else
static void * RCTProfileTrampoline = NULL ;
# endif
2015-10-20 04:11:38 -07:00
RCT_EXTERN void RCTProfileTrampolineStart ( id , SEL ) ;
void RCTProfileTrampolineStart ( id self , SEL cmd )
{
2015-12-14 13:07:24 -08:00
/ * *
* This call might be during dealloc , so we shouldn ' t retain the object in the
* block .
* /
Class klass = [ self class ] ;
2016-09-05 11:11:37 -07:00
RCT_PROFILE _BEGIN _EVENT ( RCTProfileTagAlways , ( [ NSString stringWithFormat : @ "-[%s %s]" , class_getName ( klass ) , sel_getName ( cmd ) ] ) , nil ) ;
2015-10-20 04:11:38 -07:00
}
RCT_EXTERN void RCTProfileTrampolineEnd ( void ) ;
void RCTProfileTrampolineEnd ( void )
{
2016-09-05 11:11:37 -07:00
RCT_PROFILE _END _EVENT ( RCTProfileTagAlways , @ "objc_call,modules,auto" ) ;
2015-06-05 04:23:51 -07:00
}
2016-05-04 06:54:12 -07:00
static UIView * (*originalCreateView)(RCTComponentData *, SEL, NSNumber *) ;
static UIView * RCTProfileCreateView ( RCTComponentData * self , SEL _cmd , NSNumber * tag )
{
UIView * view = originalCreateView ( self , _cmd , tag ) ;
RCTProfileHookInstance ( view ) ;
return view ;
}
static void RCTProfileHookUIManager ( RCTUIManager * uiManager )
{
dispatch_async ( dispatch_get _main _queue ( ) , ^ {
for ( id view in [ uiManager valueForKey : @ "viewRegistry" ] ) {
RCTProfileHookInstance ( [ uiManager viewForReactTag : view ] ) ;
}
Method createView = class_getInstanceMethod ( [ RCTComponentData class ] , @ selector ( createViewWithTag : ) ) ;
if ( method_getImplementation ( createView ) ! = ( IMP ) RCTProfileCreateView ) {
originalCreateView = ( typeof ( originalCreateView ) ) method_getImplementation ( createView ) ;
method_setImplementation ( createView , ( IMP ) RCTProfileCreateView ) ;
}
} ) ;
}
void RCTProfileHookInstance ( id instance )
2015-12-14 13:07:24 -08:00
{
Class moduleClass = object_getClass ( instance ) ;
/ * *
* We swizzle the instance - class method to return the original class , but
* object_getClass will return the actual class .
*
* If they are different , it means that the object is returning the original
* class , but it ' s actual class is the proxy subclass we created .
* /
if ( [ instance class ] ! = moduleClass ) {
return ;
}
Class proxyClass = objc_allocateClassPair ( moduleClass , RCTProfileProxyClassName ( moduleClass ) , 0 ) ;
if ( ! proxyClass ) {
proxyClass = objc_getClass ( RCTProfileProxyClassName ( moduleClass ) ) ;
if ( proxyClass ) {
object_setClass ( instance , proxyClass ) ;
}
return ;
}
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 ) ;
/ * *
* Bail out on struct returns ( except arm64 ) - we don ' t use it enough
* to justify writing a stret version
* /
# ifdef __arm64 __
BOOL returnsStruct = NO ;
# else
const char * typeEncoding = method_getTypeEncoding ( method ) ;
// bail out on structs and unions ( since they might contain structs )
BOOL returnsStruct = typeEncoding [ 0 ] = = ' { ' || typeEncoding [ 0 ] = = ' ( ' ;
# endif
/ * *
* Avoid hooking into NSObject methods , methods generated by React Native
* and special methods that start ` . ` ( e . g . . cxx_destruct )
* /
if ( [ NSStringFromSelector ( selector ) hasPrefix : @ "rct" ] || [ NSObject instancesRespondToSelector : selector ] || sel_getName ( selector ) [ 0 ] = = ' . ' || returnsStruct ) {
continue ;
}
const char * types = method_getTypeEncoding ( method ) ;
class_addMethod ( proxyClass , selector , ( IMP ) RCTProfileTrampoline , types ) ;
}
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 ) ) ;
}
objc_registerClassPair ( proxyClass ) ;
object_setClass ( instance , proxyClass ) ;
2016-05-04 06:54:12 -07:00
if ( moduleClass = = [ RCTUIManager class ] ) {
RCTProfileHookUIManager ( ( RCTUIManager * ) instance ) ;
}
2015-12-14 13:07:24 -08:00
}
2015-07-22 10:54:45 -07:00
void RCTProfileHookModules ( RCTBridge * bridge )
2015-06-05 04:23:51 -07:00
{
2016-03-11 06:20:24 -08:00
_RCTProfilingBridge = bridge ;
2015-10-27 12:59:27 -07:00
# pragma clang diagnostic push
# pragma clang diagnostic ignored "-Wtautological-pointer-compare"
2015-11-26 06:38:11 -08:00
if ( RCTProfileTrampoline = = NULL ) {
2015-10-20 04:11:38 -07:00
return ;
}
2015-10-27 12:59:27 -07:00
# pragma clang diagnostic pop
2015-10-20 04:11:38 -07:00
2016-05-13 17:15:05 -07:00
RCT_PROFILE _BEGIN _EVENT ( RCTProfileTagAlways , @ "RCTProfileHookModules" , nil ) ;
2015-08-07 06:42:34 -07:00
for ( RCTModuleData * moduleData in [ bridge valueForKey : @ "moduleDataByID" ] ) {
2016-05-04 06:54:12 -07:00
// Only hook modules with an instance , to prevent initializing everything
if ( [ moduleData hasInstance ] ) {
[ bridge dispatchBlock : ^ {
RCTProfileHookInstance ( moduleData . instance ) ;
} queue : moduleData . methodQueue ] ;
2015-12-14 13:07:24 -08:00
}
2016-05-04 06:54:12 -07:00
}
2016-09-05 11:11:37 -07:00
RCT_PROFILE _END _EVENT ( RCTProfileTagAlways , @ "" ) ;
2015-12-14 13:07:24 -08:00
}
2015-06-05 04:23:51 -07:00
2015-12-14 13:07:24 -08:00
static void RCTProfileUnhookInstance ( id instance )
{
if ( [ instance class ] ! = object_getClass ( instance ) ) {
object_setClass ( instance , [ instance class ] ) ;
2015-06-15 07:53:45 -07:00
}
2015-06-05 04:23:51 -07:00
}
void RCTProfileUnhookModules ( RCTBridge * bridge )
{
2016-03-11 06:20:24 -08:00
_RCTProfilingBridge = nil ;
2015-10-02 04:16:04 -07:00
dispatch_group _enter ( RCTProfileGetUnhookGroup ( ) ) ;
2016-05-04 06:54:12 -07:00
NSDictionary * moduleDataByID = [ bridge valueForKey : @ "moduleDataByID" ] ;
for ( RCTModuleData * moduleData in moduleDataByID ) {
if ( [ moduleData hasInstance ] ) {
RCTProfileUnhookInstance ( moduleData . instance ) ;
}
2015-10-02 04:16:04 -07:00
}
2016-05-04 06:54:12 -07:00
if ( [ bridge moduleIsInitialized : [ RCTUIManager class ] ] ) {
dispatch_async ( dispatch_get _main _queue ( ) , ^ {
for ( id view in [ bridge . uiManager valueForKey : @ "viewRegistry" ] ) {
RCTProfileUnhookInstance ( view ) ;
}
2015-12-14 13:07:24 -08:00
2016-05-04 06:54:12 -07:00
dispatch_group _leave ( RCTProfileGetUnhookGroup ( ) ) ;
} ) ;
}
2015-06-05 04:23:51 -07:00
}
2015-11-23 09:48:43 -08:00
# pragma mark - Private ObjC class only used for the vSYNC CADisplayLink target
@ interface RCTProfile : NSObject
@ end
@ implementation RCTProfile
2015-12-09 09:53:00 -08:00
+ ( void ) vsync : ( CADisplayLink * ) displayLink
2015-11-23 09:48:43 -08:00
{
2016-05-13 17:14:59 -07:00
RCTProfileImmediateEvent ( RCTProfileTagAlways , @ "VSYNC" , displayLink . timestamp , ' g ' ) ;
2015-11-23 09:48:43 -08:00
}
2016-03-11 06:20:24 -08:00
+ ( void ) reload
{
2016-12-07 16:27:55 -08:00
[ RCTProfilingBridge ( ) reload ] ;
2016-03-11 06:20:24 -08:00
}
+ ( void ) toggle : ( UIButton * ) target
{
BOOL isProfiling = RCTProfileIsProfiling ( ) ;
// Start and Stop are switched here , since we ' re going to toggle isProfiling
[ target setTitle : isProfiling ? @ "Start" : @ "Stop"
forState : UIControlStateNormal ] ;
if ( isProfiling ) {
RCTProfileEnd ( RCTProfilingBridge ( ) , ^ ( NSString * result ) {
NSString * outFile = [ NSTemporaryDirectory ( ) stringByAppendingString : @ "tmp_trace.json" ] ;
[ result writeToFile : outFile
atomically : YES
encoding : NSUTF8StringEncoding
error : nil ] ;
2016-09-27 06:19:45 -07:00
# if ! TARGET_OS _TV
2016-03-11 06:20:24 -08:00
UIActivityViewController * activityViewController = [ [ UIActivityViewController alloc ] initWithActivityItems : @ [ [ NSURL fileURLWithPath : outFile ] ]
applicationActivities : nil ] ;
2017-01-09 19:27:48 -08:00
activityViewController . completionWithItemsHandler = ^ ( __unused UIActivityType activityType ,
__unused BOOL completed ,
__unused NSArray * items ,
__unused NSError * error ) {
2016-03-11 06:20:24 -08:00
RCTProfileControlsWindow . hidden = NO ;
} ;
RCTProfileControlsWindow . hidden = YES ;
dispatch_async ( dispatch_get _main _queue ( ) , ^ {
2017-04-01 03:16:52 -07:00
[ [ [ [ RCTSharedApplication ( ) delegate ] window ] rootViewController ] presentViewController : activityViewController
2016-03-11 06:20:24 -08:00
animated : YES
completion : nil ] ;
} ) ;
2016-09-27 06:19:45 -07:00
# endif
2016-03-11 06:20:24 -08:00
} ) ;
} else {
RCTProfileInit ( RCTProfilingBridge ( ) ) ;
}
}
+ ( void ) drag : ( UIPanGestureRecognizer * ) gestureRecognizer
{
CGPoint translation = [ gestureRecognizer translationInView : RCTProfileControlsWindow ] ;
RCTProfileControlsWindow . center = CGPointMake (
RCTProfileControlsWindow . center . x + translation . x ,
RCTProfileControlsWindow . center . y + translation . y
) ;
[ gestureRecognizer setTranslation : CGPointMake ( 0 , 0 )
inView : RCTProfileControlsWindow ] ;
}
2015-11-23 09:48:43 -08:00
@ end
2015-06-05 04:23:51 -07:00
2015-04-20 04:55:05 -07:00
# pragma mark - Public Functions
2015-11-04 09:00:01 -08:00
dispatch_queue _t RCTProfileGetQueue ( void )
{
static dispatch_queue _t queue ;
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
queue = dispatch_queue _create ( "com.facebook.react.Profiler" , DISPATCH_QUEUE _SERIAL ) ;
} ) ;
return queue ;
}
2015-04-20 04:55:05 -07:00
BOOL RCTProfileIsProfiling ( void )
{
2017-08-01 03:51:33 -07:00
return atomic_load ( & RCTProfileProfiling ) ;
2015-04-20 04:55:05 -07:00
}
2015-06-05 04:23:51 -07:00
void RCTProfileInit ( RCTBridge * bridge )
2015-04-20 04:55:05 -07:00
{
2015-11-04 09:00:01 -08:00
// TODO : enable assert JS thread from any file ( and assert here )
2017-08-01 03:51:33 -07:00
BOOL wasProfiling = atomic_fetch _or ( & RCTProfileProfiling , 1 ) ;
if ( wasProfiling ) {
2015-11-09 08:02:46 -08:00
return ;
}
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2017-03-30 06:36:21 -07:00
systrace_buffer = callbacks -> start ( ) ;
2015-09-23 11:59:44 -07:00
} else {
2015-11-04 09:00:01 -08:00
NSTimeInterval time = CACurrentMediaTime ( ) ;
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
RCTProfileStartTime = time ;
2015-09-23 11:59:44 -07:00
RCTProfileOngoingEvents = [ NSMutableDictionary new ] ;
RCTProfileInfo = @ {
2017-07-25 04:40:19 -07:00
kProfileTraceEvents : [ NSMutableArray new ] ,
kProfileSamples : [ NSMutableArray new ] ,
2015-09-23 11:59:44 -07:00
} ;
2015-11-04 09:00:01 -08:00
} ) ;
2015-09-23 11:59:44 -07:00
}
2015-06-02 06:15:53 -07:00
2016-02-12 03:49:51 -08:00
// Set up thread ordering
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
2017-03-21 18:25:00 -07:00
NSArray * orderedThreads = @ [ @ "JS async" , @ "RCTPerformanceLogger" , @ "com.facebook.react.JavaScript" ,
@ ( RCTUIManagerQueueName ) , @ "main" ] ;
2016-02-16 12:41:20 -08:00
[ orderedThreads enumerateObjectsUsingBlock : ^ ( NSString * thread , NSUInteger idx , __unused BOOL * stop ) {
2017-07-25 04:40:19 -07:00
RCTProfileAddEvent ( kProfileTraceEvents ,
2016-02-12 03:49:51 -08:00
@ "ph" : @ "M" , // metadata event
@ "name" : @ "thread_sort_index" ,
@ "tid" : thread ,
@ "args" : @ { @ "sort_index" : @ ( -1000 + ( NSInteger ) idx ) }
) ;
} ] ;
} ) ;
2015-11-04 09:00:01 -08:00
RCTProfileHookModules ( bridge ) ;
2015-11-23 09:48:43 -08:00
RCTProfileDisplayLink = [ CADisplayLink displayLinkWithTarget : [ RCTProfile class ]
selector : @ selector ( vsync : ) ] ;
[ RCTProfileDisplayLink addToRunLoop : [ NSRunLoop mainRunLoop ]
forMode : NSRunLoopCommonModes ] ;
2015-06-02 06:15:53 -07:00
[ [ NSNotificationCenter defaultCenter ] postNotificationName : RCTProfileDidStartProfiling
2015-12-15 05:42:45 -08:00
object : bridge ] ;
2015-04-20 04:55:05 -07:00
}
2015-11-04 09:00:01 -08:00
void RCTProfileEnd ( RCTBridge * bridge , void ( ^ callback ) ( NSString * ) )
2015-04-20 04:55:05 -07:00
{
2015-11-04 09:00:01 -08:00
// assert JavaScript thread here again
2017-08-01 03:51:33 -07:00
BOOL wasProfiling = atomic_fetch _and ( & RCTProfileProfiling , 0 ) ;
if ( ! wasProfiling ) {
2015-11-04 09:00:01 -08:00
return ;
2015-11-03 07:22:28 -08:00
}
2015-06-02 06:15:53 -07:00
[ [ NSNotificationCenter defaultCenter ] postNotificationName : RCTProfileDidEndProfiling
2015-12-15 05:42:45 -08:00
object : bridge ] ;
2015-06-02 06:15:53 -07:00
2015-11-23 09:48:43 -08:00
[ RCTProfileDisplayLink invalidate ] ;
RCTProfileDisplayLink = nil ;
2015-11-04 09:00:01 -08:00
RCTProfileUnhookModules ( bridge ) ;
2015-06-05 04:23:51 -07:00
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2017-03-30 06:36:21 -07:00
if ( systrace_buffer ) {
callbacks -> stop ( ) ;
callback ( @ ( systrace_buffer ) ) ;
}
2015-09-23 11:59:44 -07:00
} else {
2015-11-04 09:00:01 -08:00
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
2015-09-23 11:59:44 -07:00
NSString * log = RCTJSONStringify ( RCTProfileInfo , NULL ) ;
RCTProfileEventID = 0 ;
RCTProfileInfo = nil ;
RCTProfileOngoingEvents = nil ;
2015-06-05 04:23:51 -07:00
2015-11-04 09:00:01 -08:00
callback ( log ) ;
} ) ;
2015-09-23 11:59:44 -07:00
}
2015-04-20 04:55:05 -07:00
}
2015-11-04 09:00:01 -08:00
static NSMutableArray < NSArray * > * RCTProfileGetThreadEvents ( NSThread * thread )
2015-04-20 04:55:05 -07:00
{
2015-08-19 23:36:11 -07:00
static NSString * const RCTProfileThreadEventsKey = @ "RCTProfileThreadEventsKey" ;
2015-11-03 14:45:46 -08:00
NSMutableArray < NSArray * > * threadEvents =
2015-11-04 09:00:01 -08:00
thread . threadDictionary [ RCTProfileThreadEventsKey ] ;
2015-08-19 23:36:11 -07:00
if ( ! threadEvents ) {
2015-11-02 07:58:47 -08:00
threadEvents = [ NSMutableArray new ] ;
2015-11-04 09:00:01 -08:00
thread . threadDictionary [ RCTProfileThreadEventsKey ] = threadEvents ;
2015-08-19 23:36:11 -07:00
}
return threadEvents ;
}
2015-11-04 09:00:01 -08:00
void _RCTProfileBeginEvent (
NSThread * calleeThread ,
NSTimeInterval time ,
uint64_t tag ,
NSString * name ,
2017-05-12 17:57:12 -07:00
NSDictionary < NSString * , NSString * > * args
2015-11-04 09:00:01 -08:00
) {
2015-08-19 23:36:11 -07:00
CHECK ( ) ;
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2017-05-15 03:36:58 -07:00
systrace_arg _t * systraceArgs = newSystraceArgsFromDictionary ( args ) ;
callbacks -> begin_section ( tag , name . UTF8String , args . count , systraceArgs ) ;
free ( systraceArgs ) ;
2015-09-23 11:59:44 -07:00
return ;
}
2016-05-13 17:14:59 -07:00
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
NSMutableArray * events = RCTProfileGetThreadEvents ( calleeThread ) ;
[ events addObject : @ [
RCTProfileTimestamp ( time ) ,
name ,
RCTNullIfNil ( args ) ,
] ] ;
} ) ;
2015-08-19 23:36:11 -07:00
}
2015-11-04 09:00:01 -08:00
void _RCTProfileEndEvent (
NSThread * calleeThread ,
NSString * threadName ,
NSTimeInterval time ,
2015-09-23 11:59:44 -07:00
uint64_t tag ,
2016-09-05 11:11:37 -07:00
NSString * category
2015-08-19 23:36:11 -07:00
) {
CHECK ( ) ;
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2016-09-05 11:11:37 -07:00
callbacks -> end_section ( tag , 0 , nil ) ;
2015-09-23 11:59:44 -07:00
return ;
}
2016-05-13 17:14:59 -07:00
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
NSMutableArray < NSArray * > * events = RCTProfileGetThreadEvents ( calleeThread ) ;
NSArray * event = events . lastObject ;
[ events removeLastObject ] ;
2015-08-19 23:36:11 -07:00
2016-05-13 17:14:59 -07:00
if ( ! event ) {
return ;
}
2015-08-19 23:36:11 -07:00
2016-05-13 17:14:59 -07:00
NSNumber * start = event [ 0 ] ;
2017-07-25 04:40:19 -07:00
RCTProfileAddEvent ( kProfileTraceEvents ,
2016-05-13 17:14:59 -07:00
@ "tid" : threadName ,
@ "name" : event [ 1 ] ,
@ "cat" : category ,
@ "ph" : @ "X" ,
@ "ts" : start ,
@ "dur" : @ ( RCTProfileTimestamp ( time ) . doubleValue - start . doubleValue ) ,
2016-09-05 11:11:37 -07:00
@ "args" : event [ 2 ] ,
2016-05-13 17:14:59 -07:00
) ;
} ) ;
2015-04-20 04:55:05 -07:00
}
2015-11-08 10:19:26 -08:00
NSUInteger RCTProfileBeginAsyncEvent (
2015-09-23 11:59:44 -07:00
uint64_t tag ,
2015-08-19 23:36:11 -07:00
NSString * name ,
2017-05-12 17:57:12 -07:00
NSDictionary < NSString * , NSString * > * args
2015-08-19 23:36:11 -07:00
) {
CHECK ( 0 ) ;
2015-11-08 10:19:26 -08:00
static NSUInteger eventID = 0 ;
2015-08-19 23:36:11 -07:00
2015-11-04 09:00:01 -08:00
NSTimeInterval time = CACurrentMediaTime ( ) ;
2015-11-08 10:19:26 -08:00
NSUInteger currentEventID = + + eventID ;
2015-11-04 09:00:01 -08:00
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2017-05-15 03:36:58 -07:00
systrace_arg _t * systraceArgs = newSystraceArgsFromDictionary ( args ) ;
callbacks -> begin_async _section ( tag , name . UTF8String , ( int ) ( currentEventID % INT_MAX ) , args . count , systraceArgs ) ;
free ( systraceArgs ) ;
2015-09-23 11:59:44 -07:00
} else {
2015-11-04 09:00:01 -08:00
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
RCTProfileOngoingEvents [ @ ( currentEventID ) ] = @ [
RCTProfileTimestamp ( time ) ,
2015-09-23 11:59:44 -07:00
name ,
RCTNullIfNil ( args ) ,
] ;
2015-11-04 09:00:01 -08:00
} ) ;
2015-09-23 11:59:44 -07:00
}
2015-08-19 23:36:11 -07:00
2015-11-04 09:00:01 -08:00
return currentEventID ;
2015-08-19 23:36:11 -07:00
}
void RCTProfileEndAsyncEvent (
2015-09-23 11:59:44 -07:00
uint64_t tag ,
2015-08-19 23:36:11 -07:00
NSString * category ,
2015-11-08 10:19:26 -08:00
NSUInteger cookie ,
2015-09-23 11:59:44 -07:00
NSString * name ,
2016-09-05 11:11:37 -07:00
NSString * threadName
2015-08-19 23:36:11 -07:00
) {
2015-04-20 04:55:05 -07:00
CHECK ( ) ;
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2016-09-05 11:11:37 -07:00
callbacks -> end_async _section ( tag , name . UTF8String , ( int ) ( cookie % INT_MAX ) , 0 , nil ) ;
2015-09-23 11:59:44 -07:00
return ;
}
2015-11-04 09:00:01 -08:00
NSTimeInterval time = CACurrentMediaTime ( ) ;
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
2015-08-19 23:36:11 -07:00
NSArray * event = RCTProfileOngoingEvents [ @ ( cookie ) ] ;
2015-11-04 09:00:01 -08:00
2015-08-19 23:36:11 -07:00
if ( event ) {
2015-11-04 09:00:01 -08:00
NSNumber * endTimestamp = RCTProfileTimestamp ( time ) ;
2015-04-22 07:03:55 -07:00
2017-07-25 04:40:19 -07:00
RCTProfileAddEvent ( kProfileTraceEvents ,
2015-11-04 09:00:01 -08:00
@ "tid" : threadName ,
2015-08-19 23:36:11 -07:00
@ "name" : event [ 1 ] ,
@ "cat" : category ,
2015-04-22 07:03:55 -07:00
@ "ph" : @ "X" ,
2015-08-19 23:36:11 -07:00
@ "ts" : event [ 0 ] ,
@ "dur" : @ ( endTimestamp . doubleValue - [ event [ 0 ] doubleValue ] ) ,
2016-09-05 11:11:37 -07:00
@ "args" : event [ 2 ] ,
2015-04-22 07:03:55 -07:00
) ;
2015-08-19 23:36:11 -07:00
[ RCTProfileOngoingEvents removeObjectForKey : @ ( cookie ) ] ;
2015-04-22 07:03:55 -07:00
}
2015-11-04 09:00:01 -08:00
} ) ;
2015-04-20 04:55:05 -07:00
}
2015-08-19 23:36:11 -07:00
void RCTProfileImmediateEvent (
2015-09-23 11:59:44 -07:00
uint64_t tag ,
2015-08-19 23:36:11 -07:00
NSString * name ,
2015-12-09 09:53:00 -08:00
NSTimeInterval time ,
2015-08-19 23:36:11 -07:00
char scope
) {
2015-04-20 04:55:05 -07:00
CHECK ( ) ;
2015-08-19 23:36:11 -07:00
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
callbacks -> instant_section ( tag , name . UTF8String , scope ) ;
return ;
}
2015-11-04 09:00:01 -08:00
NSString * threadName = RCTCurrentThreadName ( ) ;
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
2017-07-25 04:40:19 -07:00
RCTProfileAddEvent ( kProfileTraceEvents ,
2015-11-04 09:00:01 -08:00
@ "tid" : threadName ,
2015-04-22 07:03:55 -07:00
@ "name" : name ,
2015-11-04 09:00:01 -08:00
@ "ts" : RCTProfileTimestamp ( time ) ,
2015-08-19 23:36:11 -07:00
@ "scope" : @ ( scope ) ,
2015-04-22 07:03:55 -07:00
@ "ph" : @ "i" ,
2015-08-21 10:15:04 -07:00
@ "args" : RCTProfileGetMemoryUsage ( ) ,
2015-04-22 07:03:55 -07:00
) ;
2015-11-04 09:00:01 -08:00
} ) ;
2015-04-20 04:55:05 -07:00
}
2015-04-20 12:33:52 -01:00
2016-08-09 03:53:59 -07:00
NSUInteger _RCTProfileBeginFlowEvent ( void )
2015-06-03 05:38:21 -07:00
{
static NSUInteger flowID = 0 ;
2016-09-27 06:07:29 -07:00
CHECK ( 0 ) ;
2015-09-23 11:59:44 -07:00
2016-08-09 03:53:59 -07:00
NSUInteger cookie = + + flowID ;
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2016-09-27 06:07:29 -07:00
callbacks -> begin_async _flow ( 1 , "flow" , ( int ) cookie ) ;
return cookie ;
2015-09-23 11:59:44 -07:00
}
2015-11-04 09:00:01 -08:00
NSTimeInterval time = CACurrentMediaTime ( ) ;
NSString * threadName = RCTCurrentThreadName ( ) ;
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
2017-07-25 04:40:19 -07:00
RCTProfileAddEvent ( kProfileTraceEvents ,
2015-11-04 09:00:01 -08:00
@ "tid" : threadName ,
2015-10-02 04:16:04 -07:00
@ "name" : @ "flow" ,
2016-08-09 03:53:59 -07:00
@ "id" : @ ( cookie ) ,
2015-10-02 04:16:04 -07:00
@ "cat" : @ "flow" ,
@ "ph" : @ "s" ,
2015-11-04 09:00:01 -08:00
@ "ts" : RCTProfileTimestamp ( time ) ,
2015-10-02 04:16:04 -07:00
) ;
2015-06-03 05:38:21 -07:00
2015-11-04 09:00:01 -08:00
} ) ;
2016-08-09 03:53:59 -07:00
return cookie ;
2015-06-03 05:38:21 -07:00
}
2016-08-09 03:53:59 -07:00
void _RCTProfileEndFlowEvent ( NSUInteger cookie )
2015-06-03 05:38:21 -07:00
{
CHECK ( ) ;
2015-09-23 11:59:44 -07:00
if ( callbacks ! = NULL ) {
2016-09-27 06:07:29 -07:00
callbacks -> end_async _flow ( 1 , "flow" , ( int ) cookie ) ;
2015-09-23 11:59:44 -07:00
return ;
}
2015-11-04 09:00:01 -08:00
NSTimeInterval time = CACurrentMediaTime ( ) ;
NSString * threadName = RCTCurrentThreadName ( ) ;
dispatch_async ( RCTProfileGetQueue ( ) , ^ {
2017-07-25 04:40:19 -07:00
RCTProfileAddEvent ( kProfileTraceEvents ,
2015-11-04 09:00:01 -08:00
@ "tid" : threadName ,
2015-10-02 04:16:04 -07:00
@ "name" : @ "flow" ,
2016-08-09 03:53:59 -07:00
@ "id" : @ ( cookie ) ,
2015-10-02 04:16:04 -07:00
@ "cat" : @ "flow" ,
@ "ph" : @ "f" ,
2015-11-04 09:00:01 -08:00
@ "ts" : RCTProfileTimestamp ( time ) ,
2015-10-02 04:16:04 -07:00
) ;
2015-11-04 09:00:01 -08:00
} ) ;
2015-06-03 05:38:21 -07:00
}
2015-09-11 06:35:25 -07:00
void RCTProfileSendResult ( RCTBridge * bridge , NSString * route , NSData * data )
{
if ( ! [ bridge . bundleURL . scheme hasPrefix : @ "http" ] ) {
2015-12-15 03:57:26 -08:00
RCTLogWarn ( @ "Cannot upload profile information because you're not connected to the packager. The profiling data is still saved in the app container." ) ;
2015-09-11 06:35:25 -07:00
return ;
}
NSURL * URL = [ NSURL URLWithString : [ @ "/" stringByAppendingString : route ] relativeToURL : bridge . bundleURL ] ;
NSMutableURLRequest * URLRequest = [ NSMutableURLRequest requestWithURL : URL ] ;
URLRequest . HTTPMethod = @ "POST" ;
[ URLRequest setValue : @ "application/json"
forHTTPHeaderField : @ "Content-Type" ] ;
NSURLSessionTask * task =
[ [ NSURLSession sharedSession ] uploadTaskWithRequest : URLRequest
fromData : data
completionHandler :
^ ( NSData * responseData , __unused NSURLResponse * response , NSError * error ) {
if ( error ) {
RCTLogError ( @ "%@" , error . localizedDescription ) ;
} else {
NSString * message = [ [ NSString alloc ] initWithData : responseData
encoding : NSUTF8StringEncoding ] ;
if ( message . length ) {
2016-09-27 06:19:45 -07:00
# if ! TARGET_OS _TV
2015-09-11 06:35:25 -07:00
dispatch_async ( dispatch_get _main _queue ( ) , ^ {
2017-04-06 12:28:41 -07:00
UIAlertController * alertController = [ UIAlertController
alertControllerWithTitle : @ "Profile"
message : message
preferredStyle : UIAlertControllerStyleAlert ] ;
[ alertController addAction : [ UIAlertAction actionWithTitle : @ "OK"
style : UIAlertActionStyleCancel
handler : nil ] ] ;
[ RCTPresentedViewController ( ) presentViewController : alertController animated : YES completion : nil ] ;
2015-09-11 06:35:25 -07:00
} ) ;
2016-09-27 06:19:45 -07:00
# endif
2015-09-11 06:35:25 -07:00
}
}
} ] ;
[ task resume ] ;
}
2016-03-11 06:20:24 -08:00
void RCTProfileShowControls ( void )
{
static const CGFloat height = 30 ;
static const CGFloat width = 60 ;
UIWindow * window = [ [ UIWindow alloc ] initWithFrame : CGRectMake ( 20 , 80 , width * 2 , height ) ] ;
window . windowLevel = UIWindowLevelAlert + 1000 ;
window . hidden = NO ;
window . backgroundColor = [ UIColor lightGrayColor ] ;
window . layer . borderColor = [ UIColor grayColor ] . CGColor ;
window . layer . borderWidth = 1 ;
window . alpha = 0.8 ;
UIButton * startOrStop = [ [ UIButton alloc ] initWithFrame : CGRectMake ( 0 , 0 , width , height ) ] ;
[ startOrStop setTitle : RCTProfileIsProfiling ( ) ? @ "Stop" : @ "Start"
forState : UIControlStateNormal ] ;
[ startOrStop addTarget : [ RCTProfile class ] action : @ selector ( toggle : ) forControlEvents : UIControlEventTouchUpInside ] ;
startOrStop . titleLabel . font = [ UIFont systemFontOfSize : 12 ] ;
UIButton * reload = [ [ UIButton alloc ] initWithFrame : CGRectMake ( width , 0 , width , height ) ] ;
[ reload setTitle : @ "Reload" forState : UIControlStateNormal ] ;
[ reload addTarget : [ RCTProfile class ] action : @ selector ( reload ) forControlEvents : UIControlEventTouchUpInside ] ;
reload . titleLabel . font = [ UIFont systemFontOfSize : 12 ] ;
[ window addSubview : startOrStop ] ;
[ window addSubview : reload ] ;
UIPanGestureRecognizer * gestureRecognizer = [ [ UIPanGestureRecognizer alloc ] initWithTarget : [ RCTProfile class ]
action : @ selector ( drag : ) ] ;
[ window addGestureRecognizer : gestureRecognizer ] ;
RCTProfileControlsWindow = window ;
}
void RCTProfileHideControls ( void )
{
RCTProfileControlsWindow . hidden = YES ;
RCTProfileControlsWindow = nil ;
}
2015-04-20 12:33:52 -01:00
# endif