forked from TextureGroup/Texture
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathASPerformanceTestContext.mm
135 lines (116 loc) · 3.79 KB
/
ASPerformanceTestContext.mm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
//
// ASPerformanceTestContext.mm
// Texture
//
// Copyright (c) Facebook, Inc. and its affiliates. All rights reserved.
// Changes after 4/13/2017 are: Copyright (c) Pinterest, Inc. All rights reserved.
// Licensed under Apache 2.0: http://www.apache.org/licenses/LICENSE-2.0
//
#import "ASPerformanceTestContext.h"
#import <AsyncDisplayKit/ASAssert.h>
#import <Foundation/Foundation.h>
#import <QuartzCore/CABase.h>
@interface ASPerformanceTestResult ()
@property (nonatomic) NSTimeInterval timePer1000;
@property (nonatomic) NSString *caseName;
@property (nonatomic, getter=isReferenceCase) BOOL referenceCase;
@property (nonatomic) float relativePerformance;
@end
@implementation ASPerformanceTestResult
- (instancetype)init
{
self = [super init];
if (self != nil) {
_userInfo = [[NSMutableDictionary alloc] init];
}
return self;
}
- (NSString *)description
{
NSString *userInfoStr = [_userInfo.description stringByReplacingOccurrencesOfString:@"\n" withString:@" "];
return [NSString stringWithFormat:@"<%-20s: time-per-1000=%04.2f rel-perf=%04.2f user-info=%@>", _caseName.UTF8String, _timePer1000, _relativePerformance, userInfoStr];
}
@end
@implementation ASPerformanceTestContext {
NSMutableDictionary *_results;
NSInteger _iterationCount;
ASPerformanceTestResult * _Nullable _referenceResult;
}
- (instancetype)init
{
self = [super init];
if (self != nil) {
_iterationCount = 1E4;
_results = [[NSMutableDictionary alloc] init];
}
return self;
}
- (NSDictionary<NSString *, ASPerformanceTestResult *> *)results
{
return _results;
}
- (void)dealloc
{
/**
* I know this seems wacky but it's a pain to have to put this in every single test method.
*/
NSLog(@"%@", self.description);
}
- (BOOL)areAllUserInfosEqual
{
ASDisplayNodeAssert(_results.count >= 2, nil);
NSEnumerator *resultsEnumerator = [_results objectEnumerator];
NSDictionary *userInfo = [[resultsEnumerator nextObject] userInfo];
for (ASPerformanceTestResult *otherResult in resultsEnumerator) {
if ([userInfo isEqualToDictionary:otherResult.userInfo] == NO) {
return NO;
}
}
return YES;
}
- (void)addCaseWithName:(NSString *)caseName block:(AS_NOESCAPE ASTestPerformanceCaseBlock)block
{
ASDisplayNodeAssert(_results[caseName] == nil, @"Already have a case named %@", caseName);
ASPerformanceTestResult *result = [[ASPerformanceTestResult alloc] init];
result.caseName = caseName;
result.timePer1000 = [self _testPerformanceForCaseWithBlock:block] / (_iterationCount / 1000);
if (_referenceResult == nil) {
result.referenceCase = YES;
result.relativePerformance = 1.0f;
_referenceResult = result;
} else {
result.relativePerformance = _referenceResult.timePer1000 / result.timePer1000;
}
_results[caseName] = result;
}
/// Returns total work time
- (CFTimeInterval)_testPerformanceForCaseWithBlock:(AS_NOESCAPE ASTestPerformanceCaseBlock)block
{
__block CFTimeInterval time = 0;
for (NSInteger i = 0; i < _iterationCount; i++) {
__block CFTimeInterval start = 0;
__block BOOL calledStop = NO;
@autoreleasepool {
block(i, ^{
ASDisplayNodeAssert(start == 0, @"Called startMeasuring block twice.");
start = CACurrentMediaTime();
}, ^{
time += (CACurrentMediaTime() - start);
ASDisplayNodeAssert(calledStop == NO, @"Called stopMeasuring block twice.");
ASDisplayNodeAssert(start != 0, @"Failed to call startMeasuring block");
calledStop = YES;
});
}
ASDisplayNodeAssert(calledStop, @"Failed to call stopMeasuring block.");
}
return time;
}
- (NSString *)description
{
NSMutableString *str = [NSMutableString stringWithString:@"Results:\n"];
for (ASPerformanceTestResult *result in [_results objectEnumerator]) {
[str appendFormat:@"\t%@\n", result];
}
return str;
}
@end