-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathconv2d_bench.nim
170 lines (145 loc) · 5.56 KB
/
conv2d_bench.nim
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
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
# Apache v2 License
# Mamy Ratsimbazafy
# ##########################################
# Benchmarking tools
import random, times, stats, strformat, math, sequtils
proc warmup() =
# Warmup - make sure cpu is on max perf
let start = epochTime()
var foo = 123
for i in 0 ..< 300_000_000:
foo += i*i mod 456
foo = foo mod 789
# Compiler shouldn't optimize away the results as cpuTime rely on sideeffects
let stop = epochTime()
echo &"Warmup: {stop - start:>4.4f} s, result {foo} (displayed to avoid compiler optimizing warmup away)"
template printStats(name: string, output: openarray, buffer_size: int) {.dirty.} =
let bufsize = float buffer_size * sizeof(float32)
echo "\n" & name
echo &"Collected {stats.n} samples in {global_stop - global_start:>4.3f} seconds"
echo &"Average time: {stats.mean * 1000 :>4.3f} ms"
echo &"Stddev time: {stats.standardDeviationS * 1000 :>4.3f} ms"
echo &"Min time: {stats.min * 1000 :>4.3f} ms"
echo &"Max time: {stats.max * 1000 :>4.3f} ms"
echo &"Perf: {req_ops.float / stats.mean / float(10^9):>4.3f} GFLOP/s"
echo &"Buffer size: {bufsize / float(10^6):>4.3f} MB"
echo "\nDisplay output[0] to make sure it's not optimized away"
echo output[0] # Prevents compiler from optimizing stuff away
template bench(name: string, buffer_size: int, initialisation, body: untyped) {.dirty.}=
block: # Actual bench
var stats: RunningStat
let global_start = epochTime()
for _ in 0 ..< nb_samples:
initialisation
let start = epochTime()
body
let stop = epochTime()
stats.push stop - start
let global_stop = epochTime()
printStats(name, output, buffer_size)
# #############################################
# Params
import ./conv2d_common
const
N = 16
C_in = 3
C_out = 20
H = 224
W = 224
kH = 3
kW = 3
padding: Padding = (0, 0)
strides: Strides = (1, 1)
const
ishape: TensorShape = (N, C_in, H, W)
kshape: KernelShape = (C_out, C_in, kH, kW)
in_size = N * C_in * H * W
let req_ops = conv2d_required_ops(
ishape, kshape, padding, strides
)
let req_bytes = sizeof(float32) * conv2d_required_data(
ishape, kshape, padding, strides
)
let out_shape: TensorShape = conv2d_out_shape(
ishape, kshape, padding, strides
)
let out_size = out_shape.n * out_shape.c * out_shape.h * out_shape.w
# #############################################
import
./conv2d_direct_convolution,
./conv2d_im2col
proc benchDirect(input, kernel: seq[float32], nb_samples: int) =
var output = newSeq[float32](out_size)
bench("Direct convolution", 0):
# Initialisation, not measured apart for the "Collected n samples in ... seconds"
zeroMem(output[0].addr, out_size) # We zero memory between computation
do:
# Main work
conv2d_direct(output, input, ishape, kernel, kshape, padding, strides)
proc benchim2col(input, kernel: seq[float32], nb_samples: int) =
var output = newSeq[float32](out_size)
let buffer_size = im2col_workspace_size(ishape, kshape, padding, strides)
var workspace = newSeq[float32](buffer_size)
let pworkspace = workspace[0].addr
bench("Im2Col convolution", buffer_size):
# Initialisation, not measured apart for the "Collected n samples in ... seconds"
zeroMem(output[0].addr, out_size) # We zero memory between computation
do:
# Main work
conv2d_im2col(output, out_shape, input, ishape, kernel, kshape, padding, strides, pworkspace)
# ###########################################
when defined(fast_math):
{.passC:"-ffast-math".}
when defined(march_native):
{.passC:"-march=native".}
when isMainModule:
randomize(42) # For reproducibility
warmup()
echo ""
echo "Input shape: " & $ishape
echo "Kernel shape: " & $kshape
echo "Padding and strides: ", padding, strides
echo "Output shape: " & $out_shape
echo &"Required number of operations: {req_ops.float / float(10^6):>9.3f} millions"
echo &"Required bytes: {req_bytes.float / float(10^6):>9.3f} MB"
echo &"Arithmetic intensity: {req_ops.float / req_bytes.float:>9.3f} FLOP/byte"
block:
let input = newSeqWith(N*C_in*H*W, float32 rand(1.0))
let kernel = newSeqWith(C_out*C_in*kH*kW, float32 rand(1.0))
benchDirect(input, kernel, nb_samples = 20)
benchim2col(input, kernel, nb_samples = 20)
# CPU: i5-5257U https://ark.intel.com/products/84985/Intel-Core-i5-5257U-Processor-3M-Cache-up-to-3-10-GHz-
# Frequency: 2.7GHz, Turbo 3.1
# Cores: 2
# flop per cycle: 32 (AVX2 - 2*(8*add + 8*mul))
#
# Peak theoretical GFLOPS = 2.7 * 2 * 32 = 172.8
# Peak theoretical GFLOPS, turbo = 3.1 * 2 * 32 = 198.4
# Warmup: 1.1958 s, result 224 (displayed to avoid compiler optimizing warmup away)
# Input shape: (n: 16, c: 3, h: 224, w: 224)
# Kernel shape: (c_out: 20, c_in: 3, kH: 3, kW: 3)
# Padding and strides: (h: 0, w: 0)(h: 1, w: 1)
# Output shape: (n: 16, c: 20, h: 222, w: 222)
# Required number of operations: 851.628 millions
# Required bytes: 72.719 MB
# Arithmetic intensity: 11.711 FLOP/byte
# Direct convolution
# Collected 20 samples in 6.587 seconds
# Average time: 328.233 ms
# Stddev time: 17.892 ms
# Min time: 316.517 ms
# Max time: 381.466 ms
# Perf: 2.595 GFLOP/s
# Buffer size: 0.000 MB
# Display output[0] to make sure it's not optimized away
# 7.531180381774902
# Im2Col convolution
# Collected 20 samples in 0.849 seconds
# Average time: 41.317 ms
# Stddev time: 3.416 ms
# Min time: 39.624 ms
# Max time: 54.946 ms
# Perf: 20.612 GFLOP/s
# Buffer size: 5.323 MB
# Display output[0] to make sure it's not optimized away
# 7.129507064819336