ruby-prof 0.16.2 → 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (203) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGES +532 -467
  3. data/LICENSE +24 -24
  4. data/README.rdoc +5 -454
  5. data/Rakefile +110 -113
  6. data/bin/ruby-prof +380 -340
  7. data/bin/ruby-prof-check-trace +45 -45
  8. data/ext/ruby_prof/extconf.rb +36 -64
  9. data/ext/ruby_prof/rp_allocation.c +279 -0
  10. data/ext/ruby_prof/rp_allocation.h +31 -0
  11. data/ext/ruby_prof/rp_call_info.c +271 -407
  12. data/ext/ruby_prof/rp_call_info.h +35 -48
  13. data/ext/ruby_prof/rp_measure_allocations.c +52 -76
  14. data/ext/ruby_prof/rp_measure_memory.c +42 -77
  15. data/ext/ruby_prof/rp_measure_process_time.c +67 -71
  16. data/ext/ruby_prof/rp_measure_wall_time.c +62 -45
  17. data/ext/ruby_prof/rp_measurement.c +230 -0
  18. data/ext/ruby_prof/rp_measurement.h +50 -0
  19. data/ext/ruby_prof/rp_method.c +630 -411
  20. data/ext/ruby_prof/rp_method.h +70 -52
  21. data/ext/ruby_prof/rp_profile.c +895 -0
  22. data/ext/ruby_prof/rp_profile.h +37 -0
  23. data/ext/ruby_prof/rp_stack.c +196 -128
  24. data/ext/ruby_prof/rp_stack.h +56 -51
  25. data/ext/ruby_prof/rp_thread.c +337 -273
  26. data/ext/ruby_prof/rp_thread.h +36 -27
  27. data/ext/ruby_prof/ruby_prof.c +48 -671
  28. data/ext/ruby_prof/ruby_prof.h +17 -56
  29. data/ext/ruby_prof/vc/ruby_prof.sln +20 -21
  30. data/ext/ruby_prof/vc/{ruby_prof_20.vcxproj → ruby_prof.vcxproj} +38 -5
  31. data/lib/ruby-prof.rb +52 -58
  32. data/lib/ruby-prof/assets/call_stack_printer.html.erb +713 -0
  33. data/lib/ruby-prof/assets/call_stack_printer.png +0 -0
  34. data/lib/ruby-prof/assets/graph_printer.html.erb +356 -0
  35. data/lib/ruby-prof/call_info.rb +57 -126
  36. data/lib/ruby-prof/call_info_visitor.rb +38 -40
  37. data/lib/ruby-prof/compatibility.rb +109 -178
  38. data/lib/ruby-prof/exclude_common_methods.rb +198 -0
  39. data/lib/ruby-prof/measurement.rb +14 -0
  40. data/lib/ruby-prof/method_info.rb +90 -129
  41. data/lib/ruby-prof/printers/abstract_printer.rb +127 -85
  42. data/lib/ruby-prof/printers/call_info_printer.rb +51 -41
  43. data/lib/ruby-prof/printers/call_stack_printer.rb +182 -260
  44. data/lib/ruby-prof/printers/call_tree_printer.rb +151 -130
  45. data/lib/ruby-prof/printers/dot_printer.rb +132 -132
  46. data/lib/ruby-prof/printers/flat_printer.rb +52 -70
  47. data/lib/ruby-prof/printers/graph_html_printer.rb +63 -244
  48. data/lib/ruby-prof/printers/graph_printer.rb +114 -116
  49. data/lib/ruby-prof/printers/multi_printer.rb +127 -58
  50. data/lib/ruby-prof/profile.rb +33 -55
  51. data/lib/ruby-prof/rack.rb +171 -95
  52. data/lib/ruby-prof/task.rb +147 -147
  53. data/lib/ruby-prof/thread.rb +35 -41
  54. data/lib/ruby-prof/version.rb +3 -3
  55. data/lib/unprof.rb +10 -10
  56. data/ruby-prof.gemspec +58 -57
  57. data/test/abstract_printer_test.rb +26 -0
  58. data/test/alias_test.rb +129 -0
  59. data/test/basic_test.rb +129 -128
  60. data/test/call_info_visitor_test.rb +31 -31
  61. data/test/duplicate_names_test.rb +32 -32
  62. data/test/dynamic_method_test.rb +53 -55
  63. data/test/enumerable_test.rb +21 -21
  64. data/test/exceptions_test.rb +24 -16
  65. data/test/exclude_methods_test.rb +146 -0
  66. data/test/exclude_threads_test.rb +53 -53
  67. data/test/fiber_test.rb +73 -79
  68. data/test/gc_test.rb +96 -0
  69. data/test/line_number_test.rb +161 -71
  70. data/test/marshal_test.rb +119 -0
  71. data/test/measure_allocations.rb +30 -0
  72. data/test/measure_allocations_test.rb +385 -26
  73. data/test/measure_allocations_trace_test.rb +385 -0
  74. data/test/measure_memory_trace_test.rb +756 -0
  75. data/test/measure_process_time_test.rb +849 -63
  76. data/test/measure_times.rb +54 -0
  77. data/test/measure_wall_time_test.rb +459 -255
  78. data/test/multi_printer_test.rb +71 -83
  79. data/test/no_method_class_test.rb +15 -15
  80. data/test/parser_timings.rb +24 -0
  81. data/test/pause_resume_test.rb +166 -166
  82. data/test/prime.rb +56 -54
  83. data/test/printer_call_stack_test.rb +28 -0
  84. data/test/printer_call_tree_test.rb +31 -0
  85. data/test/printer_flat_test.rb +68 -0
  86. data/test/printer_graph_html_test.rb +60 -0
  87. data/test/printer_graph_test.rb +41 -0
  88. data/test/printers_test.rb +141 -255
  89. data/test/printing_recursive_graph_test.rb +81 -127
  90. data/test/rack_test.rb +157 -93
  91. data/test/recursive_test.rb +210 -215
  92. data/test/singleton_test.rb +38 -38
  93. data/test/stack_printer_test.rb +64 -78
  94. data/test/start_stop_test.rb +109 -112
  95. data/test/test_helper.rb +24 -264
  96. data/test/thread_test.rb +144 -187
  97. data/test/unique_call_path_test.rb +190 -202
  98. data/test/yarv_test.rb +56 -55
  99. metadata +34 -114
  100. data/doc/LICENSE.html +0 -114
  101. data/doc/README_rdoc.html +0 -603
  102. data/doc/Rack.html +0 -95
  103. data/doc/Rack/RubyProf.html +0 -226
  104. data/doc/RubyProf.html +0 -962
  105. data/doc/RubyProf/AbstractPrinter.html +0 -546
  106. data/doc/RubyProf/AggregateCallInfo.html +0 -551
  107. data/doc/RubyProf/CallInfo.html +0 -639
  108. data/doc/RubyProf/CallInfoPrinter.html +0 -120
  109. data/doc/RubyProf/CallInfoVisitor.html +0 -198
  110. data/doc/RubyProf/CallStackPrinter.html +0 -1121
  111. data/doc/RubyProf/CallTreePrinter.html +0 -641
  112. data/doc/RubyProf/Cmd.html +0 -631
  113. data/doc/RubyProf/DotPrinter.html +0 -257
  114. data/doc/RubyProf/FlatPrinter.html +0 -163
  115. data/doc/RubyProf/FlatPrinterWithLineNumbers.html +0 -208
  116. data/doc/RubyProf/GraphHtmlPrinter.html +0 -552
  117. data/doc/RubyProf/GraphPrinter.html +0 -139
  118. data/doc/RubyProf/MethodInfo.html +0 -745
  119. data/doc/RubyProf/MultiPrinter.html +0 -360
  120. data/doc/RubyProf/Profile.html +0 -763
  121. data/doc/RubyProf/ProfileTask.html +0 -490
  122. data/doc/RubyProf/Thread.html +0 -310
  123. data/doc/created.rid +0 -31
  124. data/doc/css/fonts.css +0 -167
  125. data/doc/css/rdoc.css +0 -590
  126. data/doc/examples/flat_txt.html +0 -138
  127. data/doc/examples/graph_html.html +0 -909
  128. data/doc/examples/graph_txt.html +0 -247
  129. data/doc/fonts/Lato-Light.ttf +0 -0
  130. data/doc/fonts/Lato-LightItalic.ttf +0 -0
  131. data/doc/fonts/Lato-Regular.ttf +0 -0
  132. data/doc/fonts/Lato-RegularItalic.ttf +0 -0
  133. data/doc/fonts/SourceCodePro-Bold.ttf +0 -0
  134. data/doc/fonts/SourceCodePro-Regular.ttf +0 -0
  135. data/doc/images/add.png +0 -0
  136. data/doc/images/arrow_up.png +0 -0
  137. data/doc/images/brick.png +0 -0
  138. data/doc/images/brick_link.png +0 -0
  139. data/doc/images/bug.png +0 -0
  140. data/doc/images/bullet_black.png +0 -0
  141. data/doc/images/bullet_toggle_minus.png +0 -0
  142. data/doc/images/bullet_toggle_plus.png +0 -0
  143. data/doc/images/date.png +0 -0
  144. data/doc/images/delete.png +0 -0
  145. data/doc/images/find.png +0 -0
  146. data/doc/images/loadingAnimation.gif +0 -0
  147. data/doc/images/macFFBgHack.png +0 -0
  148. data/doc/images/package.png +0 -0
  149. data/doc/images/page_green.png +0 -0
  150. data/doc/images/page_white_text.png +0 -0
  151. data/doc/images/page_white_width.png +0 -0
  152. data/doc/images/plugin.png +0 -0
  153. data/doc/images/ruby.png +0 -0
  154. data/doc/images/tag_blue.png +0 -0
  155. data/doc/images/tag_green.png +0 -0
  156. data/doc/images/transparent.png +0 -0
  157. data/doc/images/wrench.png +0 -0
  158. data/doc/images/wrench_orange.png +0 -0
  159. data/doc/images/zoom.png +0 -0
  160. data/doc/index.html +0 -626
  161. data/doc/js/darkfish.js +0 -161
  162. data/doc/js/jquery.js +0 -4
  163. data/doc/js/navigation.js +0 -142
  164. data/doc/js/navigation.js.gz +0 -0
  165. data/doc/js/search.js +0 -109
  166. data/doc/js/search_index.js +0 -1
  167. data/doc/js/search_index.js.gz +0 -0
  168. data/doc/js/searcher.js +0 -228
  169. data/doc/js/searcher.js.gz +0 -0
  170. data/doc/table_of_contents.html +0 -942
  171. data/examples/cachegrind.out.1 +0 -114
  172. data/examples/cachegrind.out.1.32313213 +0 -114
  173. data/examples/flat.txt +0 -50
  174. data/examples/graph.dot +0 -84
  175. data/examples/graph.html +0 -823
  176. data/examples/graph.txt +0 -139
  177. data/examples/multi.flat.txt +0 -23
  178. data/examples/multi.graph.html +0 -760
  179. data/examples/multi.grind.dat +0 -114
  180. data/examples/multi.stack.html +0 -547
  181. data/examples/stack.html +0 -547
  182. data/ext/ruby_prof/rp_measure.c +0 -40
  183. data/ext/ruby_prof/rp_measure.h +0 -45
  184. data/ext/ruby_prof/rp_measure_cpu_time.c +0 -136
  185. data/ext/ruby_prof/rp_measure_gc_runs.c +0 -73
  186. data/ext/ruby_prof/rp_measure_gc_time.c +0 -60
  187. data/ext/ruby_prof/vc/ruby_prof_18.vcxproj +0 -108
  188. data/ext/ruby_prof/vc/ruby_prof_19.vcxproj +0 -110
  189. data/lib/ruby-prof/aggregate_call_info.rb +0 -76
  190. data/lib/ruby-prof/assets/call_stack_printer.css.html +0 -117
  191. data/lib/ruby-prof/assets/call_stack_printer.js.html +0 -385
  192. data/lib/ruby-prof/printers/flat_printer_with_line_numbers.rb +0 -64
  193. data/test/aggregate_test.rb +0 -136
  194. data/test/block_test.rb +0 -74
  195. data/test/call_info_test.rb +0 -78
  196. data/test/issue137_test.rb +0 -63
  197. data/test/measure_cpu_time_test.rb +0 -213
  198. data/test/measure_gc_runs_test.rb +0 -32
  199. data/test/measure_gc_time_test.rb +0 -36
  200. data/test/measure_memory_test.rb +0 -33
  201. data/test/method_elimination_test.rb +0 -84
  202. data/test/module_test.rb +0 -45
  203. data/test/stack_test.rb +0 -138
@@ -1,116 +1,114 @@
1
- # encoding: utf-8
2
-
3
- module RubyProf
4
- # Generates graph[link:files/examples/graph_txt.html] profile reports as text.
5
- # To use the graph printer:
6
- #
7
- # result = RubyProf.profile do
8
- # [code to profile]
9
- # end
10
- #
11
- # printer = RubyProf::GraphPrinter.new(result)
12
- # printer.print(STDOUT, {})
13
- #
14
- # The constructor takes two arguments. See the README
15
-
16
- class GraphPrinter < AbstractPrinter
17
- PERCENTAGE_WIDTH = 8
18
- TIME_WIDTH = 11
19
- CALL_WIDTH = 17
20
-
21
- private
22
-
23
- def print_header(thread)
24
- @output << "Measure Mode: %s\n" % RubyProf.measure_mode_string
25
- @output << "Thread ID: #{thread.id}\n"
26
- @output << "Fiber ID: #{thread.fiber_id}\n" unless thread.id == thread.fiber_id
27
- @output << "Total Time: #{thread.total_time}\n"
28
- @output << "Sort by: #{sort_method}\n"
29
- @output << "\n"
30
-
31
- # 1 is for % sign
32
- @output << sprintf("%#{PERCENTAGE_WIDTH}s", "%total")
33
- @output << sprintf("%#{PERCENTAGE_WIDTH}s", "%self")
34
- @output << sprintf("%#{TIME_WIDTH}s", "total")
35
- @output << sprintf("%#{TIME_WIDTH}s", "self")
36
- @output << sprintf("%#{TIME_WIDTH}s", "wait")
37
- @output << sprintf("%#{TIME_WIDTH}s", "child")
38
- @output << sprintf("%#{CALL_WIDTH}s", "calls")
39
- @output << " name"
40
- @output << "\n"
41
- end
42
-
43
- def print_methods(thread)
44
- total_time = thread.total_time
45
- # Sort methods from longest to shortest total time
46
- methods = thread.methods.sort_by(&sort_method)
47
-
48
- # Print each method in total time order
49
- methods.reverse_each do |method|
50
- total_percentage = (method.total_time/total_time) * 100
51
- next if total_percentage < min_percent
52
-
53
- self_percentage = (method.self_time/total_time) * 100
54
-
55
- @output << "-" * 80 << "\n"
56
- print_parents(thread, method)
57
-
58
- # 1 is for % sign
59
- @output << sprintf("%#{PERCENTAGE_WIDTH-1}.2f\%", total_percentage)
60
- @output << sprintf("%#{PERCENTAGE_WIDTH-1}.2f\%", self_percentage)
61
- @output << sprintf("%#{TIME_WIDTH}.3f", method.total_time)
62
- @output << sprintf("%#{TIME_WIDTH}.3f", method.self_time)
63
- @output << sprintf("%#{TIME_WIDTH}.3f", method.wait_time)
64
- @output << sprintf("%#{TIME_WIDTH}.3f", method.children_time)
65
- @output << sprintf("%#{CALL_WIDTH}i", method.called)
66
- @output << sprintf(" %s", method.recursive? ? "*" : " ")
67
- @output << sprintf("%s", method_name(method))
68
- if print_file
69
- @output << sprintf(" %s:%s", method.source_file, method.line)
70
- end
71
- @output << "\n"
72
-
73
- print_children(method)
74
- end
75
- end
76
-
77
- def print_parents(thread, method)
78
- method.aggregate_parents.sort_by(&:total_time).each do |caller|
79
- next unless caller.parent
80
- @output << " " * 2 * PERCENTAGE_WIDTH
81
- @output << sprintf("%#{TIME_WIDTH}.3f", caller.total_time)
82
- @output << sprintf("%#{TIME_WIDTH}.3f", caller.self_time)
83
- @output << sprintf("%#{TIME_WIDTH}.3f", caller.wait_time)
84
- @output << sprintf("%#{TIME_WIDTH}.3f", caller.children_time)
85
-
86
- call_called = "#{caller.called}/#{method.called}"
87
- @output << sprintf("%#{CALL_WIDTH}s", call_called)
88
- @output << sprintf(" %s", caller.parent.target.full_name)
89
- @output << "\n"
90
- end
91
- end
92
-
93
- def print_children(method)
94
- method.aggregate_children.sort_by(&:total_time).reverse.each do |child|
95
- # Get children method
96
-
97
- @output << " " * 2 * PERCENTAGE_WIDTH
98
-
99
- @output << sprintf("%#{TIME_WIDTH}.3f", child.total_time)
100
- @output << sprintf("%#{TIME_WIDTH}.3f", child.self_time)
101
- @output << sprintf("%#{TIME_WIDTH}.3f", child.wait_time)
102
- @output << sprintf("%#{TIME_WIDTH}.3f", child.children_time)
103
-
104
- call_called = "#{child.called}/#{child.target.called}"
105
- @output << sprintf("%#{CALL_WIDTH}s", call_called)
106
- @output << sprintf(" %s", child.target.full_name)
107
- @output << "\n"
108
- end
109
- end
110
-
111
- def print_footer(thread)
112
- @output << "\n"
113
- @output << "* indicates recursively called methods\n"
114
- end
115
- end
116
- end
1
+ # encoding: utf-8
2
+
3
+ module RubyProf
4
+ # Generates graph[link:files/examples/graph_txt.html] profile reports as text.
5
+ # To use the graph printer:
6
+ #
7
+ # result = RubyProf.profile do
8
+ # [code to profile]
9
+ # end
10
+ #
11
+ # printer = RubyProf::GraphPrinter.new(result)
12
+ # printer.print(STDOUT, {})
13
+ #
14
+ # The constructor takes two arguments. See the README
15
+
16
+ class GraphPrinter < AbstractPrinter
17
+ PERCENTAGE_WIDTH = 8
18
+ TIME_WIDTH = 11
19
+ CALL_WIDTH = 17
20
+
21
+ private
22
+
23
+ def sort_method
24
+ @options[:sort_method] || :total_time
25
+ end
26
+
27
+ def print_header(thread)
28
+ @output << "Measure Mode: %s\n" % RubyProf.measure_mode_string
29
+ @output << "Thread ID: #{thread.id}\n"
30
+ @output << "Fiber ID: #{thread.fiber_id}\n"
31
+ @output << "Total Time: #{thread.total_time}\n"
32
+ @output << "Sort by: #{sort_method}\n"
33
+ @output << "\n"
34
+
35
+ # 1 is for % sign
36
+ @output << sprintf("%#{PERCENTAGE_WIDTH}s", "%total")
37
+ @output << sprintf("%#{PERCENTAGE_WIDTH}s", "%self")
38
+ @output << sprintf("%#{TIME_WIDTH}s", "total")
39
+ @output << sprintf("%#{TIME_WIDTH}s", "self")
40
+ @output << sprintf("%#{TIME_WIDTH}s", "wait")
41
+ @output << sprintf("%#{TIME_WIDTH}s", "child")
42
+ @output << sprintf("%#{CALL_WIDTH}s", "calls")
43
+ @output << " name"
44
+ @output << " location"
45
+ @output << "\n"
46
+ end
47
+
48
+ def print_methods(thread)
49
+ total_time = thread.total_time
50
+ # Sort methods from longest to shortest total time
51
+ methods = thread.methods.sort_by(&sort_method)
52
+
53
+ # Print each method in total time order
54
+ methods.reverse_each do |method|
55
+ total_percentage = (method.total_time/total_time) * 100
56
+ next if total_percentage < min_percent
57
+
58
+ self_percentage = (method.self_time/total_time) * 100
59
+
60
+ @output << "-" * 150 << "\n"
61
+ print_parents(thread, method)
62
+
63
+ # 1 is for % sign
64
+ @output << sprintf("%#{PERCENTAGE_WIDTH-1}.2f%%", total_percentage)
65
+ @output << sprintf("%#{PERCENTAGE_WIDTH-1}.2f%%", self_percentage)
66
+ @output << sprintf("%#{TIME_WIDTH}.3f", method.total_time)
67
+ @output << sprintf("%#{TIME_WIDTH}.3f", method.self_time)
68
+ @output << sprintf("%#{TIME_WIDTH}.3f", method.wait_time)
69
+ @output << sprintf("%#{TIME_WIDTH}.3f", method.children_time)
70
+ @output << sprintf("%#{CALL_WIDTH}i", method.called)
71
+ @output << sprintf(" %s", method.recursive? ? "*" : " ")
72
+ @output << sprintf("%-30s", method.full_name)
73
+ @output << sprintf(" %s", method_location(method))
74
+ @output << "\n"
75
+
76
+ print_children(method)
77
+ end
78
+ end
79
+
80
+ def print_parents(thread, method)
81
+ return if method.root?
82
+ method.callers.sort_by(&:total_time).each do |caller|
83
+ @output << " " * 2 * PERCENTAGE_WIDTH
84
+ @output << sprintf("%#{TIME_WIDTH}.3f", caller.total_time)
85
+ @output << sprintf("%#{TIME_WIDTH}.3f", caller.self_time)
86
+ @output << sprintf("%#{TIME_WIDTH}.3f", caller.wait_time)
87
+ @output << sprintf("%#{TIME_WIDTH}.3f", caller.children_time)
88
+
89
+ call_called = "#{caller.called}/#{method.called}"
90
+ @output << sprintf("%#{CALL_WIDTH}s", call_called)
91
+ @output << sprintf(" %s", caller.parent.full_name)
92
+ @output << "\n"
93
+ end
94
+ end
95
+
96
+ def print_children(method)
97
+ method.callees.sort_by(&:total_time).reverse.each do |child|
98
+ # Get children method
99
+
100
+ @output << " " * 2 * PERCENTAGE_WIDTH
101
+
102
+ @output << sprintf("%#{TIME_WIDTH}.3f", child.total_time)
103
+ @output << sprintf("%#{TIME_WIDTH}.3f", child.self_time)
104
+ @output << sprintf("%#{TIME_WIDTH}.3f", child.wait_time)
105
+ @output << sprintf("%#{TIME_WIDTH}.3f", child.children_time)
106
+
107
+ call_called = "#{child.called}/#{child.target.called}"
108
+ @output << sprintf("%#{CALL_WIDTH}s", call_called)
109
+ @output << sprintf(" %s", child.target.full_name)
110
+ @output << "\n"
111
+ end
112
+ end
113
+ end
114
+ end
@@ -1,58 +1,127 @@
1
- # encoding: utf-8
2
-
3
- module RubyProf
4
- # Helper class to simplify printing profiles of several types from
5
- # one profiling run. Currently prints a flat profile, a callgrind
6
- # profile, a call stack profile and a graph profile.
7
- class MultiPrinter
8
- def initialize(result)
9
- @stack_printer = CallStackPrinter.new(result)
10
- @graph_printer = GraphHtmlPrinter.new(result)
11
- @tree_printer = CallTreePrinter.new(result)
12
- @flat_printer = FlatPrinter.new(result)
13
- end
14
-
15
- # create profile files under options[:path] or the current
16
- # directory. options[:profile] is used as the base name for the
17
- # pofile file, defaults to "profile".
18
- def print(options)
19
- @profile = options.delete(:profile) || "profile"
20
- @directory = options.delete(:path) || File.expand_path(".")
21
-
22
- File.open(stack_profile, "w") do |f|
23
- @stack_printer.print(f, options.merge(:graph => "#{@profile}.graph.html"))
24
- end
25
-
26
- File.open(graph_profile, "w") do |f|
27
- @graph_printer.print(f, options)
28
- end
29
-
30
- @tree_printer.print(options.merge(:path => @directory, :profile => @profile))
31
-
32
- File.open(flat_profile, "w") do |f|
33
- @flat_printer.print(f, options)
34
- end
35
- end
36
-
37
- # the name of the call stack profile file
38
- def stack_profile
39
- "#{@directory}/#{@profile}.stack.html"
40
- end
41
-
42
- # the name of the graph profile file
43
- def graph_profile
44
- "#{@directory}/#{@profile}.graph.html"
45
- end
46
-
47
- # the name of the callgrind profile file
48
- def tree_profile
49
- "#{@directory}/#{@profile}.callgrind.out.#{$$}"
50
- end
51
-
52
- # the name of the flat profile file
53
- def flat_profile
54
- "#{@directory}/#{@profile}.flat.txt"
55
- end
56
-
57
- end
58
- end
1
+ # encoding: utf-8
2
+
3
+ module RubyProf
4
+ # Helper class to simplify printing profiles of several types from
5
+ # one profiling run. Currently prints a flat profile, a callgrind
6
+ # profile, a call stack profile and a graph profile.
7
+ class MultiPrinter
8
+ def initialize(result, printers = [:flat, :graph_html])
9
+ @flat_printer = FlatPrinter.new(result) if printers.include?(:flat)
10
+
11
+ @graph_printer = GraphPrinter.new(result) if printers.include?(:graph)
12
+ @graph_html_printer = GraphHtmlPrinter.new(result) if printers.include?(:graph_html)
13
+
14
+ @tree_printer = CallTreePrinter.new(result) if printers.include?(:tree)
15
+ @call_info_printer = CallInfoPrinter.new(result) if printers.include?(:call_info)
16
+
17
+ @stack_printer = CallStackPrinter.new(result) if printers.include?(:stack)
18
+ @dot_printer = DotPrinter.new(result) if printers.include?(:dot)
19
+ end
20
+
21
+ def self.needs_dir?
22
+ true
23
+ end
24
+
25
+ # create profile files under options[:path] or the current
26
+ # directory. options[:profile] is used as the base name for the
27
+ # profile file, defaults to "profile".
28
+ def print(options)
29
+ validate_print_params(options)
30
+
31
+ @profile = options.delete(:profile) || "profile"
32
+ @directory = options.delete(:path) || File.expand_path(".")
33
+
34
+ print_to_flat(options) if @flat_printer
35
+
36
+ print_to_graph(options) if @graph_printer
37
+ print_to_graph_html(options) if @graph_html_printer
38
+
39
+ print_to_stack(options) if @stack_printer
40
+ print_to_call_info(options) if @call_info_printer
41
+ print_to_tree(options) if @tree_printer
42
+ print_to_dot(options) if @dot_printer
43
+ end
44
+
45
+ # the name of the flat profile file
46
+ def flat_report
47
+ "#{@directory}/#{@profile}.flat.txt"
48
+ end
49
+
50
+ # the name of the graph profile file
51
+ def graph_report
52
+ "#{@directory}/#{@profile}.graph.txt"
53
+ end
54
+
55
+ def graph_html_report
56
+ "#{@directory}/#{@profile}.graph.html"
57
+ end
58
+
59
+ # the name of the callinfo profile file
60
+ def call_info_report
61
+ "#{@directory}/#{@profile}.call_info.txt"
62
+ end
63
+
64
+ # the name of the callgrind profile file
65
+ def tree_report
66
+ "#{@directory}/#{@profile}.callgrind.out.#{$$}"
67
+ end
68
+
69
+ # the name of the call stack profile file
70
+ def stack_report
71
+ "#{@directory}/#{@profile}.stack.html"
72
+ end
73
+
74
+ # the name of the call stack profile file
75
+ def dot_report
76
+ "#{@directory}/#{@profile}.dot"
77
+ end
78
+
79
+ def print_to_flat(options)
80
+ File.open(flat_report, "wb") do |file|
81
+ @flat_printer.print(file, options)
82
+ end
83
+ end
84
+
85
+ def print_to_graph(options)
86
+ File.open(graph_report, "wb") do |file|
87
+ @graph_printer.print(file, options)
88
+ end
89
+ end
90
+
91
+ def print_to_graph_html(options)
92
+ File.open(graph_html_report, "wb") do |file|
93
+ @graph_html_printer.print(file, options)
94
+ end
95
+ end
96
+
97
+ def print_to_call_info(options)
98
+ File.open(call_info_report, "wb") do |file|
99
+ @call_info_printer.print(file, options)
100
+ end
101
+ end
102
+
103
+ def print_to_tree(options)
104
+ @tree_printer.print(options.merge(:path => @directory, :profile => @profile))
105
+ end
106
+
107
+ def print_to_stack(options)
108
+ File.open(stack_report, "wb") do |file|
109
+ @stack_printer.print(file, options.merge(:graph => "#{@profile}.graph.html"))
110
+ end
111
+ end
112
+
113
+ def print_to_dot(options)
114
+ File.open(dot_report, "wb") do |file|
115
+ @dot_printer.print(file, options)
116
+ end
117
+ end
118
+
119
+ def validate_print_params(options)
120
+ if options.is_a?(IO)
121
+ raise ArgumentError, "#{self.class.name}#print cannot print to IO objects"
122
+ elsif !options.is_a?(Hash)
123
+ raise ArgumentError, "#{self.class.name}#print requires an options hash"
124
+ end
125
+ end
126
+ end
127
+ end
@@ -1,55 +1,33 @@
1
- # encoding: utf-8
2
-
3
- module RubyProf
4
- class Profile
5
- # This method gets called once profiling has been completed
6
- # but before results are returned to the user. Thus it provides
7
- # a hook to do any necessary post-processing on the call graph.
8
- def post_process
9
- self.threads.each do |thread|
10
- thread.detect_recursion
11
- end
12
- end
13
-
14
- # eliminate some calls from the graph by merging the information into callers.
15
- # matchers can be a list of strings or regular expressions or the name of a file containing regexps.
16
- def eliminate_methods!(matchers)
17
- matchers = read_regexps_from_file(matchers) if matchers.is_a?(String)
18
- eliminated = []
19
- threads.each do |thread|
20
- matchers.each{ |matcher| eliminated.concat(eliminate_methods(thread.methods, matcher)) }
21
- end
22
- eliminated
23
- end
24
-
25
- private
26
-
27
- # read regexps from file
28
- def read_regexps_from_file(file_name)
29
- matchers = []
30
- File.open(file_name).each_line do |l|
31
- next if (l =~ /^(#.*|\s*)$/) # emtpy lines and lines starting with #
32
- matchers << Regexp.new(l.strip)
33
- end
34
- end
35
-
36
- # eliminate methods matching matcher
37
- def eliminate_methods(methods, matcher)
38
- eliminated = []
39
- i = 0
40
- while i < methods.size
41
- method_info = methods[i]
42
- method_name = method_info.full_name
43
- if matcher === method_name
44
- raise "can't eliminate root method" if method_info.root?
45
- eliminated << methods.delete_at(i)
46
- method_info.eliminate!
47
- else
48
- i += 1
49
- end
50
- end
51
- eliminated
52
- end
53
-
54
- end
55
- end
1
+ # encoding: utf-8
2
+
3
+ require 'ruby-prof/exclude_common_methods'
4
+
5
+ module RubyProf
6
+ class Profile
7
+ # :nodoc:
8
+ def measure_mode_string
9
+ case self.measure_mode
10
+ when WALL_TIME then "wall_time"
11
+ when PROCESS_TIME then "process_time"
12
+ when ALLOCATIONS then "allocations"
13
+ when MEMORY then "memory"
14
+ end
15
+ end
16
+
17
+ # Hides methods that, when represented as a call graph, have
18
+ # extremely large in and out degrees and make navigation impossible.
19
+ def exclude_common_methods!
20
+ ExcludeCommonMethods.apply!(self)
21
+ end
22
+
23
+ def exclude_methods!(mod, *method_or_methods)
24
+ [method_or_methods].flatten.each do |name|
25
+ exclude_method!(mod, name)
26
+ end
27
+ end
28
+
29
+ def exclude_singleton_methods!(mod, *method_or_methods)
30
+ exclude_methods!(mod.singleton_class, *method_or_methods)
31
+ end
32
+ end
33
+ end