README
1 ________________________________________________________________________
2
3 PYBENCH - A Python Benchmark Suite
4 ________________________________________________________________________
5
6 Extendable suite of low-level benchmarks for measuring
7 the performance of the Python implementation
8 (interpreter, compiler or VM).
9
10 pybench is a collection of tests that provides a standardized way to
11 measure the performance of Python implementations. It takes a very
12 close look at different aspects of Python programs and let's you
13 decide which factors are more important to you than others, rather
14 than wrapping everything up in one number, like the other performance
15 tests do (e.g. pystone which is included in the Python Standard
16 Library).
17
18 pybench has been used in the past by several Python developers to
19 track down performance bottlenecks or to demonstrate the impact of
20 optimizations and new features in Python.
21
22 The command line interface for pybench is the file pybench.py. Run
23 this script with option '--help' to get a listing of the possible
24 options. Without options, pybench will simply execute the benchmark
25 and then print out a report to stdout.
26
27
28 Micro-Manual
29 ------------
30
31 Run 'pybench.py -h' to see the help screen. Run 'pybench.py' to run
32 the benchmark suite using default settings and 'pybench.py -f <file>'
33 to have it store the results in a file too.
34
35 It is usually a good idea to run pybench.py multiple times to see
36 whether the environment, timers and benchmark run-times are suitable
37 for doing benchmark tests.
38
39 You can use the comparison feature of pybench.py ('pybench.py -c
40 <file>') to check how well the system behaves in comparison to a
41 reference run.
42
43 If the differences are well below 10% for each test, then you have a
44 system that is good for doing benchmark testings. Of you get random
45 differences of more than 10% or significant differences between the
46 values for minimum and average time, then you likely have some
47 background processes running which cause the readings to become
48 inconsistent. Examples include: web-browsers, email clients, RSS
49 readers, music players, backup programs, etc.
50
51 If you are only interested in a few tests of the whole suite, you can
52 use the filtering option, e.g. 'pybench.py -t string' will only
53 run/show the tests that have 'string' in their name.
54
55 This is the current output of pybench.py --help:
56
57 """
58 ------------------------------------------------------------------------
59 PYBENCH - a benchmark test suite for Python interpreters/compilers.
60 ------------------------------------------------------------------------
61
62 Synopsis:
63 pybench.py [option] files...
64
65 Options and default settings:
66 -n arg number of rounds (10)
67 -f arg save benchmark to file arg ()
68 -c arg compare benchmark with the one in file arg ()
69 -s arg show benchmark in file arg, then exit ()
70 -w arg set warp factor to arg (10)
71 -t arg run only tests with names matching arg ()
72 -C arg set the number of calibration runs to arg (20)
73 -d hide noise in comparisons (0)
74 -v verbose output (not recommended) (0)
75 --with-gc enable garbage collection (0)
76 --with-syscheck use default sys check interval (0)
77 --timer arg use given timer (time.time)
78 -h show this help text
79 --help show this help text
80 --debug enable debugging
81 --copyright show copyright
82 --examples show examples of usage
83
84 Version:
85 2.0
86
87 The normal operation is to run the suite and display the
88 results. Use -f to save them for later reuse or comparisons.
89
90 Available timers:
91
92 time.time
93 time.clock
94 systimes.processtime
95
96 Examples:
97
98 python2.1 pybench.py -f p21.pybench
99 python2.5 pybench.py -f p25.pybench
100 python pybench.py -s p25.pybench -c p21.pybench
101 """
102
103 License
104 -------
105
106 See LICENSE file.
107
108
109 Sample output
110 -------------
111
112 """
113 -------------------------------------------------------------------------------
114 PYBENCH 2.0
115 -------------------------------------------------------------------------------
116 * using Python 2.4.2
117 * disabled garbage collection
118 * system check interval set to maximum: 2147483647
119 * using timer: time.time
120
121 Calibrating tests. Please wait...
122
123 Running 10 round(s) of the suite at warp factor 10:
124
125 * Round 1 done in 6.388 seconds.
126 * Round 2 done in 6.485 seconds.
127 * Round 3 done in 6.786 seconds.
128 ...
129 * Round 10 done in 6.546 seconds.
130
131 -------------------------------------------------------------------------------
132 Benchmark: 2006-06-12 12:09:25
133 -------------------------------------------------------------------------------
134
135 Rounds: 10
136 Warp: 10
137 Timer: time.time
138
139 Machine Details:
140 Platform ID: Linux-2.6.8-24.19-default-x86_64-with-SuSE-9.2-x86-64
141 Processor: x86_64
142
143 Python:
144 Executable: /usr/local/bin/python
145 Version: 2.4.2
146 Compiler: GCC 3.3.4 (pre 3.3.5 20040809)
147 Bits: 64bit
148 Build: Oct 1 2005 15:24:35 (#1)
149 Unicode: UCS2
150
151
152 Test minimum average operation overhead
153 -------------------------------------------------------------------------------
154 BuiltinFunctionCalls: 126ms 145ms 0.28us 0.274ms
155 BuiltinMethodLookup: 124ms 130ms 0.12us 0.316ms
156 CompareFloats: 109ms 110ms 0.09us 0.361ms
157 CompareFloatsIntegers: 100ms 104ms 0.12us 0.271ms
158 CompareIntegers: 137ms 138ms 0.08us 0.542ms
159 CompareInternedStrings: 124ms 127ms 0.08us 1.367ms
160 CompareLongs: 100ms 104ms 0.10us 0.316ms
161 CompareStrings: 111ms 115ms 0.12us 0.929ms
162 CompareUnicode: 108ms 128ms 0.17us 0.693ms
163 ConcatStrings: 142ms 155ms 0.31us 0.562ms
164 ConcatUnicode: 119ms 127ms 0.42us 0.384ms
165 CreateInstances: 123ms 128ms 1.14us 0.367ms
166 CreateNewInstances: 121ms 126ms 1.49us 0.335ms
167 CreateStringsWithConcat: 130ms 135ms 0.14us 0.916ms
168 CreateUnicodeWithConcat: 130ms 135ms 0.34us 0.361ms
169 DictCreation: 108ms 109ms 0.27us 0.361ms
170 DictWithFloatKeys: 149ms 153ms 0.17us 0.678ms
171 DictWithIntegerKeys: 124ms 126ms 0.11us 0.915ms
172 DictWithStringKeys: 114ms 117ms 0.10us 0.905ms
173 ForLoops: 110ms 111ms 4.46us 0.063ms
174 IfThenElse: 118ms 119ms 0.09us 0.685ms
175 ListSlicing: 116ms 120ms 8.59us 0.103ms
176 NestedForLoops: 125ms 137ms 0.09us 0.019ms
177 NormalClassAttribute: 124ms 136ms 0.11us 0.457ms
178 NormalInstanceAttribute: 110ms 117ms 0.10us 0.454ms
179 PythonFunctionCalls: 107ms 113ms 0.34us 0.271ms
180 PythonMethodCalls: 140ms 149ms 0.66us 0.141ms
181 Recursion: 156ms 166ms 3.32us 0.452ms
182 SecondImport: 112ms 118ms 1.18us 0.180ms
183 SecondPackageImport: 118ms 127ms 1.27us 0.180ms
184 SecondSubmoduleImport: 140ms 151ms 1.51us 0.180ms
185 SimpleComplexArithmetic: 128ms 139ms 0.16us 0.361ms
186 SimpleDictManipulation: 134ms 136ms 0.11us 0.452ms
187 SimpleFloatArithmetic: 110ms 113ms 0.09us 0.571ms
188 SimpleIntFloatArithmetic: 106ms 111ms 0.08us 0.548ms
189 SimpleIntegerArithmetic: 106ms 109ms 0.08us 0.544ms
190 SimpleListManipulation: 103ms 113ms 0.10us 0.587ms
191 SimpleLongArithmetic: 112ms 118ms 0.18us 0.271ms
192 SmallLists: 105ms 116ms 0.17us 0.366ms
193 SmallTuples: 108ms 128ms 0.24us 0.406ms
194 SpecialClassAttribute: 119ms 136ms 0.11us 0.453ms
195 SpecialInstanceAttribute: 143ms 155ms 0.13us 0.454ms
196 StringMappings: 115ms 121ms 0.48us 0.405ms
197 StringPredicates: 120ms 129ms 0.18us 2.064ms
198 StringSlicing: 111ms 127ms 0.23us 0.781ms
199 TryExcept: 125ms 126ms 0.06us 0.681ms
200 TryRaiseExcept: 133ms 137ms 2.14us 0.361ms
201 TupleSlicing: 117ms 120ms 0.46us 0.066ms
202 UnicodeMappings: 156ms 160ms 4.44us 0.429ms
203 UnicodePredicates: 117ms 121ms 0.22us 2.487ms
204 UnicodeProperties: 115ms 153ms 0.38us 2.070ms
205 UnicodeSlicing: 126ms 129ms 0.26us 0.689ms
206 -------------------------------------------------------------------------------
207 Totals: 6283ms 6673ms
208 """
209 ________________________________________________________________________
210
211 Writing New Tests
212 ________________________________________________________________________
213
214 pybench tests are simple modules defining one or more pybench.Test
215 subclasses.
216
217 Writing a test essentially boils down to providing two methods:
218 .test() which runs .rounds number of .operations test operations each
219 and .calibrate() which does the same except that it doesn't actually
220 execute the operations.
221
222
223 Here's an example:
224 ------------------
225
226 from pybench import Test
227
228 class IntegerCounting(Test):
229
230 # Version number of the test as float (x.yy); this is important
231 # for comparisons of benchmark runs - tests with unequal version
232 # number will not get compared.
233 version = 1.0
234
235 # The number of abstract operations done in each round of the
236 # test. An operation is the basic unit of what you want to
237 # measure. The benchmark will output the amount of run-time per
238 # operation. Note that in order to raise the measured timings
239 # significantly above noise level, it is often required to repeat
240 # sets of operations more than once per test round. The measured
241 # overhead per test round should be less than 1 second.
242 operations = 20
243
244 # Number of rounds to execute per test run. This should be
245 # adjusted to a figure that results in a test run-time of between
246 # 1-2 seconds (at warp 1).
247 rounds = 100000
248
249 def test(self):
250
251 """ Run the test.
252
253 The test needs to run self.rounds executing
254 self.operations number of operations each.
255
256 """
257 # Init the test
258 a = 1
259
260 # Run test rounds
261 #
262 # NOTE: Use xrange() for all test loops unless you want to face
263 # a 20MB process !
264 #
265 for i in xrange(self.rounds):
266
267 # Repeat the operations per round to raise the run-time
268 # per operation significantly above the noise level of the
269 # for-loop overhead.
270
271 # Execute 20 operations (a += 1):
272 a += 1
273 a += 1
274 a += 1
275 a += 1
276 a += 1
277 a += 1
278 a += 1
279 a += 1
280 a += 1
281 a += 1
282 a += 1
283 a += 1
284 a += 1
285 a += 1
286 a += 1
287 a += 1
288 a += 1
289 a += 1
290 a += 1
291 a += 1
292
293 def calibrate(self):
294
295 """ Calibrate the test.
296
297 This method should execute everything that is needed to
298 setup and run the test - except for the actual operations
299 that you intend to measure. pybench uses this method to
300 measure the test implementation overhead.
301
302 """
303 # Init the test
304 a = 1
305
306 # Run test rounds (without actually doing any operation)
307 for i in xrange(self.rounds):
308
309 # Skip the actual execution of the operations, since we
310 # only want to measure the test's administration overhead.
311 pass
312
313 Registering a new test module
314 -----------------------------
315
316 To register a test module with pybench, the classes need to be
317 imported into the pybench.Setup module. pybench will then scan all the
318 symbols defined in that module for subclasses of pybench.Test and
319 automatically add them to the benchmark suite.
320
321
322 Breaking Comparability
323 ----------------------
324
325 If a change is made to any individual test that means it is no
326 longer strictly comparable with previous runs, the '.version' class
327 variable should be updated. Therefafter, comparisons with previous
328 versions of the test will list as "n/a" to reflect the change.
329
330
331 Version History
332 ---------------
333
334 2.0: rewrote parts of pybench which resulted in more repeatable
335 timings:
336 - made timer a parameter
337 - changed the platform default timer to use high-resolution
338 timers rather than process timers (which have a much lower
339 resolution)
340 - added option to select timer
341 - added process time timer (using systimes.py)
342 - changed to use min() as timing estimator (average
343 is still taken as well to provide an idea of the difference)
344 - garbage collection is turned off per default
345 - sys check interval is set to the highest possible value
346 - calibration is now a separate step and done using
347 a different strategy that allows measuring the test
348 overhead more accurately
349 - modified the tests to each give a run-time of between
350 100-200ms using warp 10
351 - changed default warp factor to 10 (from 20)
352 - compared results with timeit.py and confirmed measurements
353 - bumped all test versions to 2.0
354 - updated platform.py to the latest version
355 - changed the output format a bit to make it look
356 nicer
357 - refactored the APIs somewhat
358 1.3+: Steve Holden added the NewInstances test and the filtering
359 option during the NeedForSpeed sprint; this also triggered a long
360 discussion on how to improve benchmark timing and finally
361 resulted in the release of 2.0
362 1.3: initial checkin into the Python SVN repository
363
364
365 Have fun,
366 --
367 Marc-Andre Lemburg
368 mal (a] lemburg.com
369