-
Notifications
You must be signed in to change notification settings - Fork 48
/
Copy pathBenchmark.swift
134 lines (116 loc) · 3.11 KB
/
Benchmark.swift
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
import _StringProcessing
import Foundation
public protocol RegexBenchmark {
var name: String { get }
func run()
func debug()
}
public struct Benchmark: RegexBenchmark {
public let name: String
let regex: Regex<AnyRegexOutput>
let type: MatchType
let target: String
public enum MatchType {
case whole
case first
case allMatches
}
public func run() {
switch type {
case .whole: blackHole(target.wholeMatch(of: regex))
case .allMatches: blackHole(target.matches(of: regex))
case .first: blackHole(target.firstMatch(of: regex))
}
}
}
public struct NSBenchmark: RegexBenchmark {
public let name: String
let regex: NSRegularExpression
let type: NSMatchType
let target: String
var range: NSRange {
NSRange(target.startIndex..<target.endIndex, in: target)
}
public enum NSMatchType {
case allMatches
case first
}
public func run() {
switch type {
case .allMatches: blackHole(regex.matches(in: target, range: range))
case .first: blackHole(regex.firstMatch(in: target, range: range))
}
}
}
/// A benchmark meant to be ran across multiple engines
struct CrossBenchmark {
/// The base name of the benchmark
var baseName: String
/// The string to compile in differnet engines
var regex: String
/// The text to search
var input: String
// TODO: var output, for validation
/// Whether this is whole string matching or a searching benchmark
///
/// TODO: Probably better ot have a whole-line vs search anywhere, maybe
/// accomodate multi-line matching, etc.
var isWhole: Bool = false
/// Whether or not to do firstMatch as well or just allMatches
var includeFirst: Bool = false
func register(_ runner: inout BenchmarkRunner) {
let swiftRegex = try! Regex(regex)
let nsRegex: NSRegularExpression
if isWhole {
nsRegex = try! NSRegularExpression(pattern: "^" + regex + "$")
} else {
nsRegex = try! NSRegularExpression(pattern: regex)
}
if isWhole {
runner.register(
Benchmark(
name: baseName + "Whole",
regex: swiftRegex,
type: .whole,
target: input))
runner.register(
NSBenchmark(
name: baseName + "Whole_NS",
regex: nsRegex,
type: .first,
target: input))
} else {
runner.register(
Benchmark(
name: baseName + "All",
regex: swiftRegex,
type: .allMatches,
target: input))
runner.register(
NSBenchmark(
name: baseName + "All_NS",
regex: nsRegex,
type: .allMatches,
target: input))
if includeFirst {
runner.register(
Benchmark(
name: baseName + "First",
regex: swiftRegex,
type: .first,
target: input))
runner.register(
NSBenchmark(
name: baseName + "First_NS",
regex: nsRegex,
type: .first,
target: input))
}
}
}
}
// TODO: Capture-containing benchmarks
// nom nom nom, consume the argument
@inline(never)
public func blackHole<T>(_ x: T) {
}