2015-06-19 14:59:42 -07:00
|
|
|
/**
|
|
|
|
* 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 <QuartzCore/QuartzCore.h>
|
|
|
|
|
|
|
|
#import "RCTPerformanceLogger.h"
|
|
|
|
#import "RCTRootView.h"
|
2015-10-13 03:43:43 -07:00
|
|
|
#import "RCTLog.h"
|
2016-03-15 05:40:09 -07:00
|
|
|
#import "RCTProfile.h"
|
2015-06-19 14:59:42 -07:00
|
|
|
|
|
|
|
static int64_t RCTPLData[RCTPLSize][2] = {};
|
2016-03-21 03:20:49 -07:00
|
|
|
static NSUInteger RCTPLCookies[RCTPLSize] = {};
|
2015-06-19 14:59:42 -07:00
|
|
|
|
|
|
|
void RCTPerformanceLoggerStart(RCTPLTag tag)
|
|
|
|
{
|
2016-03-15 05:40:09 -07:00
|
|
|
if (RCTProfileIsProfiling()) {
|
|
|
|
NSString *label = RCTPerformanceLoggerLabels()[tag];
|
2016-05-13 17:14:59 -07:00
|
|
|
RCTPLCookies[tag] = RCTProfileBeginAsyncEvent(RCTProfileTagAlways, label, nil);
|
2016-03-15 05:40:09 -07:00
|
|
|
}
|
|
|
|
|
2015-06-19 14:59:42 -07:00
|
|
|
RCTPLData[tag][0] = CACurrentMediaTime() * 1000;
|
2015-10-13 03:43:43 -07:00
|
|
|
RCTPLData[tag][1] = 0;
|
2015-06-19 14:59:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void RCTPerformanceLoggerEnd(RCTPLTag tag)
|
|
|
|
{
|
2015-10-13 03:43:43 -07:00
|
|
|
if (RCTPLData[tag][0] != 0 && RCTPLData[tag][1] == 0) {
|
|
|
|
RCTPLData[tag][1] = CACurrentMediaTime() * 1000;
|
2016-03-15 05:40:09 -07:00
|
|
|
|
|
|
|
if (RCTProfileIsProfiling()) {
|
|
|
|
NSString *label = RCTPerformanceLoggerLabels()[tag];
|
2016-05-13 17:14:59 -07:00
|
|
|
RCTProfileEndAsyncEvent(RCTProfileTagAlways, @"native", RCTPLCookies[tag], label, @"RCTPerformanceLogger", nil);
|
2016-03-15 05:40:09 -07:00
|
|
|
}
|
2015-10-13 03:43:43 -07:00
|
|
|
} else {
|
|
|
|
RCTLogInfo(@"Unbalanced calls start/end for tag %li", (unsigned long)tag);
|
|
|
|
}
|
2015-06-19 14:59:42 -07:00
|
|
|
}
|
|
|
|
|
2015-10-21 10:53:35 -07:00
|
|
|
void RCTPerformanceLoggerSet(RCTPLTag tag, int64_t value)
|
|
|
|
{
|
|
|
|
RCTPLData[tag][0] = 0;
|
|
|
|
RCTPLData[tag][1] = value;
|
|
|
|
}
|
|
|
|
|
2016-04-01 07:24:40 -07:00
|
|
|
void RCTPerformanceLoggerAdd(RCTPLTag tag, int64_t value)
|
|
|
|
{
|
|
|
|
RCTPLData[tag][0] = 0;
|
|
|
|
RCTPLData[tag][1] += value;
|
|
|
|
}
|
|
|
|
|
2016-03-15 05:40:09 -07:00
|
|
|
void RCTPerformanceLoggerAppendStart(RCTPLTag tag)
|
|
|
|
{
|
|
|
|
RCTPLData[tag][0] = CACurrentMediaTime() * 1000;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RCTPerformanceLoggerAppendEnd(RCTPLTag tag)
|
|
|
|
{
|
|
|
|
if (RCTPLData[tag][0] != 0) {
|
|
|
|
RCTPLData[tag][1] += CACurrentMediaTime() * 1000 - RCTPLData[tag][0];
|
|
|
|
RCTPLData[tag][0] = 0;
|
|
|
|
} else {
|
|
|
|
RCTLogInfo(@"Unbalanced calls start/end for tag %li", (unsigned long)tag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-03 14:45:46 -08:00
|
|
|
NSArray<NSNumber *> *RCTPerformanceLoggerOutput(void)
|
2015-06-19 14:59:42 -07:00
|
|
|
{
|
2016-04-01 07:24:40 -07:00
|
|
|
NSMutableArray *result = [NSMutableArray array];
|
|
|
|
for (NSUInteger index = 0; index < RCTPLSize; index++) {
|
|
|
|
[result addObject:@(RCTPLData[index][0])];
|
|
|
|
[result addObject:@(RCTPLData[index][1])];
|
|
|
|
}
|
|
|
|
return result;
|
2015-06-19 14:59:42 -07:00
|
|
|
}
|
|
|
|
|
2015-10-30 16:51:48 -07:00
|
|
|
NSArray *RCTPerformanceLoggerLabels(void)
|
|
|
|
{
|
2016-03-15 05:40:09 -07:00
|
|
|
static NSArray *labels;
|
|
|
|
static dispatch_once_t token;
|
|
|
|
dispatch_once(&token, ^{
|
|
|
|
labels = @[
|
|
|
|
@"ScriptDownload",
|
|
|
|
@"ScriptExecution",
|
2016-04-01 07:24:40 -07:00
|
|
|
@"RAMBundleLoad",
|
|
|
|
@"RAMStartupCodeSize",
|
2016-04-01 11:08:19 -07:00
|
|
|
@"RAMNativeRequires",
|
2016-04-01 07:24:40 -07:00
|
|
|
@"RAMNativeRequiresCount",
|
|
|
|
@"RAMNativeRequiresSize",
|
2016-03-15 05:40:09 -07:00
|
|
|
@"NativeModuleInit",
|
|
|
|
@"NativeModuleMainThread",
|
|
|
|
@"NativeModulePrepareConfig",
|
|
|
|
@"NativeModuleInjectConfig",
|
2016-03-17 07:26:54 -07:00
|
|
|
@"NativeModuleMainThreadUsesCount",
|
2016-03-15 05:40:09 -07:00
|
|
|
@"JSCExecutorSetup",
|
2016-04-01 07:24:40 -07:00
|
|
|
@"BridgeStartup",
|
2016-03-15 05:40:09 -07:00
|
|
|
@"RootViewTTI",
|
|
|
|
@"BundleSize",
|
|
|
|
];
|
|
|
|
});
|
|
|
|
return labels;
|
2015-10-30 16:51:48 -07:00
|
|
|
}
|
|
|
|
|
2015-06-19 14:59:42 -07:00
|
|
|
@interface RCTPerformanceLogger : NSObject <RCTBridgeModule>
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation RCTPerformanceLogger
|
|
|
|
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
|
|
|
@synthesize bridge = _bridge;
|
|
|
|
|
2016-05-04 07:06:09 -07:00
|
|
|
- (instancetype)init
|
|
|
|
{
|
|
|
|
// We're only overriding this to ensure the module gets created at startup
|
|
|
|
// TODO (t11106126): Remove once we have more declarative control over module setup.
|
|
|
|
return [super init];
|
|
|
|
}
|
|
|
|
|
2015-11-25 03:09:00 -08:00
|
|
|
- (void)setBridge:(RCTBridge *)bridge
|
2015-06-19 14:59:42 -07:00
|
|
|
{
|
2015-11-25 03:09:00 -08:00
|
|
|
_bridge = bridge;
|
|
|
|
|
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
|
|
selector:@selector(sendTimespans)
|
|
|
|
name:RCTContentDidAppearNotification
|
|
|
|
object:nil];
|
2015-06-19 14:59:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc
|
|
|
|
{
|
|
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)sendTimespans
|
|
|
|
{
|
2015-07-09 07:26:54 -07:00
|
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
|
|
|
2015-06-19 14:59:42 -07:00
|
|
|
[_bridge enqueueJSCall:@"PerformanceLogger.addTimespans" args:@[
|
|
|
|
RCTPerformanceLoggerOutput(),
|
2015-10-30 16:51:48 -07:00
|
|
|
RCTPerformanceLoggerLabels(),
|
2015-06-19 14:59:42 -07:00
|
|
|
]];
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|