forked from TextureGroup/Texture
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathASPerformanceTestContext.m
126 lines (110 loc) · 3.53 KB
/
ASPerformanceTestContext.m
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
//
// ASPerformanceTestContext.m
// Texture
//
// Created by Adlai Holler on 8/28/16.
// Copyright © 2016 Facebook. All rights reserved.
//
#import <Foundation/Foundation.h>
#import "ASPerformanceTestContext.h"
#import <AsyncDisplayKit/ASAssert.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 dictionary];
}
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;
}
- (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