1 ________________________________________________________________________ 2 3 PYBENCH - A Python Benchmark Suite 4 ________________________________________________________________________ 5 6 Extendable suite of 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