require 'benchmark_driver/version' require 'benchmark' require 'tempfile' class BenchmarkDriver MEASURE_TYPES = %w[loop_count ips] DEFAULT_LOOP_COUNT = 100_000 DEFAULT_IPS_DURATION = 1 # @param [String] measure_type - "loop_count"|"ips" # @param [Integer] measure_num - Loop count for "loop_type", duration seconds for "ips" # @param [Array] execs - ["path1", "path2"] or `["ruby1::path1", "ruby2::path2"]` # @param [Boolean] verbose def initialize(measure_type: 'loop_count', measure_num: nil, execs: ['ruby'], verbose: false) unless MEASURE_TYPES.include?(measure_type) abort "unsupported measure type: #{measure_type.dump}" end @measure_type = measure_type @measure_num = measure_num @execs = execs.map do |exec| name, path = exec.split('::', 2) Executable.new(name, path || name) end @verbose = verbose end # @param [Hash] root_hash def run(root_hash) root = BenchmarkRoot.new(Hash[root_hash.map { |k, v| [k.to_sym, v] }]) results = root.benchmarks.map do |benchmark| metrics_by_exec = {} iterations = calc_iterations(@execs.first, benchmark) @execs.each do |exec| if @verbose puts "--- Running #{benchmark.name.dump} with #{exec.name.dump} #{iterations} times ---" puts "#{benchmark.benchmark_script(iterations)}\n" end elapsed_time = run_benchmark(exec, benchmark, iterations) metrics_by_exec[exec] = BenchmarkMetrics.new(iterations, elapsed_time) end BenchmarkResult.new(benchmark.name, metrics_by_exec) end puts if @verbose case @measure_type when 'loop_count' LoopCountReporter.report(@execs, results) when 'ips' IpsReporter.report(@execs, results) else raise "unexpected measure type: #{@measure_type.dump}" end end private # Estimate iterations to finish benchmark within `@duration`. def calc_iterations(exec, benchmark) case @measure_type when 'loop_count' @measure_num || benchmark.loop_count || DEFAULT_LOOP_COUNT when 'ips' # TODO: Change to try from 1, 10, 100 ... base = 1000 time = run_benchmark(exec, benchmark, base) duration = @measure_num || DEFAULT_IPS_DURATION (duration / time * base).to_i else raise "unexpected measure type: #{@measure_type.dump}" end end def run_benchmark(exec, benchmark, iterations) # TODO: raise error if negative measure_script(exec.path, benchmark.benchmark_script(iterations)) - measure_script(exec.path, benchmark.overhead_script(iterations)) end def measure_script(ruby, script) Tempfile.create do |f| f.write(script) f.close cmd = "#{ruby} #{f.path}" Benchmark.measure { system(cmd, out: File::NULL) }.real end end class BenchmarkRoot # @param [String] name # @param [String] prelude # @param [Integer,nil] loop_count # @param [String,nil] benchmark - For running single instant benchmark # @param [Array] benchmarks - For running multiple benchmarks def initialize(name:, prelude: '', loop_count: nil, benchmark: nil, benchmarks: []) if benchmark unless benchmarks.empty? raise ArgumentError.new("Only either :benchmark or :benchmarks can be specified") end @benchmarks = [BenchmarkScript.new(name: name, prelude: prelude, benchmark: benchmark)] else @benchmarks = benchmarks.map do |hash| BenchmarkScript.new(Hash[hash.map { |k, v| [k.to_sym, v] }]).tap do |b| b.inherit_root(prelude: prelude, loop_count: loop_count) end end end end # @return [Array] attr_reader :benchmarks end class BenchmarkScript # @param [String] name # @param [String] prelude # @param [String] benchmark def initialize(name:, prelude: '', loop_count: nil, benchmark:) @name = name @prelude = prelude @loop_count = loop_count @benchmark = benchmark end # @return [String] attr_reader :name # @return [Integer] attr_reader :loop_count def inherit_root(prelude:, loop_count:) @prelude = "#{prelude}\n#{@prelude}" if @loop_count.nil? && loop_count @loop_count = loop_count end end def overhead_script(iterations) <<-RUBY #{@prelude} i = 0 while i < #{iterations} i += 1 end RUBY end def benchmark_script(iterations) <<-RUBY #{@prelude} i = 0 while i < #{iterations} i += 1 #{@benchmark} end RUBY end end class BenchmarkResult < Struct.new( :name, # @param [String] :metrics_by_exec, # @param [Hash{ Executable => BenchmarkMetrics }] ) def iterations_of(exec) metrics_by_exec.fetch(exec).iterations end def elapsed_time_of(exec) metrics_by_exec.fetch(exec).elapsed_time end def ips_of(exec) iterations_of(exec) / elapsed_time_of(exec) end end class BenchmarkMetrics < Struct.new( :iterations, # @param [Integer] :elapsed_time, # @param [Float] - Elapsed time in seconds ) end class Executable < Struct.new( :name, # @param [String] :path, # @param [String] ) end module LoopCountReporter class << self # @param [Array] execs # @param [Array] results def report(execs, results) puts "benchmark results:" puts "Execution time (sec)" puts "#{'%-16s' % 'name'} #{execs.map { |e| "%-8s" % e.name }.join(' ')}" results.each do |result| print '%-16s ' % result.name puts execs.map { |exec| "%-8s" % ("%.3f" % result.elapsed_time_of(exec)) }.join(' ') end puts if execs.size > 1 report_speedup(execs, results) end end private def report_speedup(execs, results) compared = execs.first rest = execs - [compared] puts "Speedup ratio: compare with the result of `#{compared.name}' (greater is better)" puts "#{'%-16s' % 'name'} #{rest.map { |e| "%-8s" % e.name }.join(' ')}" results.each do |result| print '%-16s ' % result.name puts rest.map { |exec| "%-8s" % ("%.3f" % (result.ips_of(exec) / result.ips_of(compared))) }.join(' ') end puts end end end module IpsReporter class << self # @param [Array] execs # @param [Array] results def report(execs, results) puts "Result -------------------------------------------" puts "#{' ' * 16} #{execs.map { |e| "%13s" % e.name }.join(' ')}" results.each do |result| print '%16s ' % result.name puts execs.map { |exec| "%13s" % ("%.1f i/s" % result.ips_of(exec)) }.join(' ') end puts if execs.size > 1 compare(execs, results) end end private def compare(execs, results) results.each do |result| puts "Comparison: #{result.name}" sorted = execs.sort_by { |e| -result.ips_of(e) } first = sorted.first sorted.each do |exec| if exec == first puts "%16s: %12s i/s" % [first.name, "%.1f" % result.ips_of(first)] else puts "%16s: %12s i/s - %.2fx slower" % [exec.name, "%.1f" % result.ips_of(exec), result.ips_of(first) / result.ips_of(exec)] end end puts end end end end end