ruby-prof 1.1.0-x64-mingw32 → 1.4.2-x64-mingw32

Sign up to get free protection for your applications and to get access to all the features.
Files changed (91) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGES +48 -1
  3. data/Rakefile +2 -14
  4. data/bin/ruby-prof +100 -152
  5. data/ext/ruby_prof/extconf.rb +8 -28
  6. data/ext/ruby_prof/rp_aggregate_call_tree.c +59 -0
  7. data/ext/ruby_prof/rp_aggregate_call_tree.h +13 -0
  8. data/ext/ruby_prof/rp_allocation.c +67 -59
  9. data/ext/ruby_prof/rp_allocation.h +3 -3
  10. data/ext/ruby_prof/rp_call_tree.c +369 -0
  11. data/ext/ruby_prof/rp_call_tree.h +43 -0
  12. data/ext/ruby_prof/rp_call_trees.c +288 -0
  13. data/ext/ruby_prof/rp_call_trees.h +28 -0
  14. data/ext/ruby_prof/rp_measure_allocations.c +12 -14
  15. data/ext/ruby_prof/rp_measure_process_time.c +12 -14
  16. data/ext/ruby_prof/rp_measure_wall_time.c +17 -15
  17. data/ext/ruby_prof/rp_measurement.c +47 -40
  18. data/ext/ruby_prof/rp_measurement.h +7 -7
  19. data/ext/ruby_prof/rp_method.c +116 -255
  20. data/ext/ruby_prof/rp_method.h +31 -39
  21. data/ext/ruby_prof/rp_profile.c +316 -303
  22. data/ext/ruby_prof/rp_profile.h +1 -3
  23. data/ext/ruby_prof/rp_stack.c +122 -106
  24. data/ext/ruby_prof/rp_stack.h +17 -20
  25. data/ext/ruby_prof/rp_thread.c +136 -111
  26. data/ext/ruby_prof/rp_thread.h +12 -9
  27. data/ext/ruby_prof/ruby_prof.c +27 -23
  28. data/ext/ruby_prof/ruby_prof.h +9 -0
  29. data/ext/ruby_prof/vc/ruby_prof.sln +8 -0
  30. data/ext/ruby_prof/vc/ruby_prof.vcxproj +22 -7
  31. data/lib/2.7/ruby_prof.so +0 -0
  32. data/lib/ruby-prof.rb +5 -5
  33. data/lib/ruby-prof/assets/call_stack_printer.html.erb +4 -7
  34. data/lib/ruby-prof/assets/graph_printer.html.erb +5 -6
  35. data/lib/ruby-prof/{call_info.rb → call_tree.rb} +6 -6
  36. data/lib/ruby-prof/call_tree_visitor.rb +36 -0
  37. data/lib/ruby-prof/compatibility.rb +0 -10
  38. data/lib/ruby-prof/measurement.rb +5 -2
  39. data/lib/ruby-prof/method_info.rb +3 -15
  40. data/lib/ruby-prof/printers/abstract_printer.rb +12 -2
  41. data/lib/ruby-prof/printers/call_info_printer.rb +12 -10
  42. data/lib/ruby-prof/printers/call_stack_printer.rb +20 -22
  43. data/lib/ruby-prof/printers/call_tree_printer.rb +1 -1
  44. data/lib/ruby-prof/printers/dot_printer.rb +3 -3
  45. data/lib/ruby-prof/printers/flat_printer.rb +3 -2
  46. data/lib/ruby-prof/printers/graph_printer.rb +4 -5
  47. data/lib/ruby-prof/printers/multi_printer.rb +2 -2
  48. data/lib/ruby-prof/profile.rb +8 -4
  49. data/lib/ruby-prof/rack.rb +51 -127
  50. data/lib/ruby-prof/thread.rb +3 -18
  51. data/lib/ruby-prof/version.rb +1 -1
  52. data/ruby-prof.gemspec +7 -0
  53. data/test/alias_test.rb +42 -45
  54. data/test/basic_test.rb +0 -86
  55. data/test/{call_info_visitor_test.rb → call_tree_visitor_test.rb} +6 -5
  56. data/test/call_trees_test.rb +66 -0
  57. data/test/exclude_methods_test.rb +17 -12
  58. data/test/fiber_test.rb +95 -39
  59. data/test/gc_test.rb +36 -42
  60. data/test/inverse_call_tree_test.rb +175 -0
  61. data/test/line_number_test.rb +67 -70
  62. data/test/marshal_test.rb +7 -13
  63. data/test/measure_allocations_test.rb +224 -234
  64. data/test/measure_allocations_trace_test.rb +224 -234
  65. data/test/measure_memory_trace_test.rb +814 -469
  66. data/test/measure_process_time_test.rb +0 -64
  67. data/test/measure_times.rb +2 -0
  68. data/test/measure_wall_time_test.rb +34 -58
  69. data/test/pause_resume_test.rb +19 -10
  70. data/test/prime.rb +1 -3
  71. data/test/prime_script.rb +6 -0
  72. data/test/printer_call_stack_test.rb +0 -1
  73. data/test/printer_call_tree_test.rb +0 -1
  74. data/test/printer_flat_test.rb +61 -30
  75. data/test/printer_graph_html_test.rb +0 -1
  76. data/test/printer_graph_test.rb +3 -4
  77. data/test/printers_test.rb +2 -2
  78. data/test/printing_recursive_graph_test.rb +1 -1
  79. data/test/profile_test.rb +16 -0
  80. data/test/rack_test.rb +0 -64
  81. data/test/recursive_test.rb +50 -54
  82. data/test/start_stop_test.rb +19 -19
  83. data/test/test_helper.rb +6 -17
  84. data/test/thread_test.rb +11 -11
  85. data/test/unique_call_path_test.rb +25 -95
  86. metadata +22 -11
  87. data/ext/ruby_prof/rp_call_info.c +0 -271
  88. data/ext/ruby_prof/rp_call_info.h +0 -35
  89. data/lib/2.6.5/ruby_prof.so +0 -0
  90. data/lib/ruby-prof/call_info_visitor.rb +0 -38
  91. data/test/parser_timings.rb +0 -24
@@ -1,7 +1,7 @@
1
1
  /* Copyright (C) 2005-2019 Shugo Maeda <shugo@ruby-lang.org> and Charlie Savage <cfis@savagexi.com>
2
2
  Please see the LICENSE file for copyright and distribution information */
3
3
 
4
- /* :nodoc: */
4
+ /* :nodoc: */
5
5
  #include "rp_measurement.h"
6
6
 
7
7
  #if defined(__APPLE__)
@@ -12,11 +12,12 @@
12
12
 
13
13
  static VALUE cMeasureWallTime;
14
14
 
15
- static double
16
- measure_wall_time(rb_trace_arg_t* trace_arg)
15
+ static double measure_wall_time(rb_trace_arg_t* trace_arg)
17
16
  {
18
17
  #if defined(_WIN32)
19
- return GetTickCount();
18
+ LARGE_INTEGER time;
19
+ QueryPerformanceCounter(&time);
20
+ return (double)time.QuadPart;
20
21
  #elif defined(__APPLE__)
21
22
  return mach_absolute_time();// * (uint64_t)mach_timebase.numer / (uint64_t)mach_timebase.denom;
22
23
  #elif defined(__linux__)
@@ -26,18 +27,19 @@ measure_wall_time(rb_trace_arg_t* trace_arg)
26
27
  #else
27
28
  struct timeval tv;
28
29
  gettimeofday(&tv, NULL);
29
- return tv.tv_sec + (tv.tv_usec / 1000000.0);
30
+ return tv.tv_sec + (tv.tv_usec / 1000000.0);
30
31
  #endif
31
32
  }
32
33
 
33
- static double
34
- multiplier_wall_time(void)
34
+ static double multiplier_wall_time(void)
35
35
  {
36
36
  #if defined(_WIN32)
37
- return 1.0/1000.0;
37
+ LARGE_INTEGER frequency;
38
+ QueryPerformanceFrequency(&frequency);
39
+ return 1.0 / frequency.QuadPart;
38
40
  #elif defined(__APPLE__)
39
41
  mach_timebase_info_data_t mach_timebase;
40
- mach_timebase_info (&mach_timebase);
42
+ mach_timebase_info(&mach_timebase);
41
43
  return (uint64_t)mach_timebase.numer / (uint64_t)mach_timebase.denom / 1000000000.0;
42
44
  #else
43
45
  return 1.0;
@@ -46,12 +48,12 @@ multiplier_wall_time(void)
46
48
 
47
49
  prof_measurer_t* prof_measurer_wall_time(bool track_allocations)
48
50
  {
49
- prof_measurer_t* measure = ALLOC(prof_measurer_t);
50
- measure->mode = MEASURE_WALL_TIME;
51
- measure->measure = measure_wall_time;
52
- measure->multiplier = multiplier_wall_time();
53
- measure->track_allocations = track_allocations;
54
- return measure;
51
+ prof_measurer_t* measure = ALLOC(prof_measurer_t);
52
+ measure->mode = MEASURE_WALL_TIME;
53
+ measure->measure = measure_wall_time;
54
+ measure->multiplier = multiplier_wall_time();
55
+ measure->track_allocations = track_allocations;
56
+ return measure;
55
57
  }
56
58
 
57
59
  void rp_init_measure_wall_time()
@@ -40,7 +40,7 @@ double prof_measure(prof_measurer_t* measurer, rb_trace_arg_t* trace_arg)
40
40
  }
41
41
 
42
42
  /* ======= prof_measurement_t ========*/
43
- prof_measurement_t *prof_measurement_create(void)
43
+ prof_measurement_t* prof_measurement_create(void)
44
44
  {
45
45
  prof_measurement_t* result = ALLOC(prof_measurement_t);
46
46
  result->total_time = 0;
@@ -51,67 +51,78 @@ prof_measurement_t *prof_measurement_create(void)
51
51
  return result;
52
52
  }
53
53
 
54
- static void
55
- prof_measurement_ruby_gc_free(void *data)
54
+ void prof_measurement_mark(void* data)
56
55
  {
57
- prof_measurement_t* measurement = (prof_measurement_t*)data;
56
+ if (!data) return;
58
57
 
58
+ prof_measurement_t* measurement_data = (prof_measurement_t*)data;
59
+
60
+ if (measurement_data->object != Qnil)
61
+ rb_gc_mark(measurement_data->object);
62
+ }
63
+
64
+ static void prof_measurement_ruby_gc_free(void* data)
65
+ {
66
+ if (data)
67
+ {
68
+ // Measurements are freed by their owning object (call info or method)
69
+ prof_measurement_t* measurement = (prof_measurement_t*)data;
70
+ measurement->object = Qnil;
71
+ }
72
+ }
73
+
74
+ void prof_measurement_free(prof_measurement_t* measurement)
75
+ {
59
76
  /* Has this measurement object been accessed by Ruby? If
60
77
  yes clean it up so to avoid a segmentation fault. */
61
78
  if (measurement->object != Qnil)
62
79
  {
63
- RDATA(measurement->object)->dmark = NULL;
64
- RDATA(measurement->object)->dfree = NULL;
65
- RDATA(measurement->object)->data = NULL;
80
+ RTYPEDDATA(measurement->object)->data = NULL;
66
81
  measurement->object = Qnil;
67
82
  }
68
- }
69
83
 
70
- void
71
- prof_measurement_free(prof_measurement_t* measurement)
72
- {
73
- prof_measurement_ruby_gc_free(measurement);
74
84
  xfree(measurement);
75
85
  }
76
86
 
77
- size_t
78
- prof_measurement_size(const void *data)
87
+ size_t prof_measurement_size(const void* data)
79
88
  {
80
89
  return sizeof(prof_measurement_t);
81
90
  }
82
91
 
83
- void
84
- prof_measurement_mark(void *data)
92
+ static const rb_data_type_t measurement_type =
85
93
  {
86
- prof_measurement_t* measurement = (prof_measurement_t*)data;
87
- if (measurement->object != Qnil)
88
- rb_gc_mark(measurement->object);
89
- }
94
+ .wrap_struct_name = "Measurement",
95
+ .function =
96
+ {
97
+ .dmark = prof_measurement_mark,
98
+ .dfree = prof_measurement_ruby_gc_free,
99
+ .dsize = prof_measurement_size,
100
+ },
101
+ .data = NULL,
102
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY
103
+ };
90
104
 
91
- VALUE
92
- prof_measurement_wrap(prof_measurement_t* measurement)
105
+ VALUE prof_measurement_wrap(prof_measurement_t* measurement)
93
106
  {
94
107
  if (measurement->object == Qnil)
95
108
  {
96
- measurement->object = Data_Wrap_Struct(cRpMeasurement, prof_measurement_mark, prof_measurement_ruby_gc_free, measurement);
109
+ measurement->object = TypedData_Wrap_Struct(cRpMeasurement, &measurement_type, measurement);
97
110
  }
98
111
  return measurement->object;
99
112
  }
100
113
 
101
- static VALUE
102
- prof_measurement_allocate(VALUE klass)
114
+ static VALUE prof_measurement_allocate(VALUE klass)
103
115
  {
104
- prof_measurement_t *measurement = prof_measurement_create();
116
+ prof_measurement_t* measurement = prof_measurement_create();
105
117
  measurement->object = prof_measurement_wrap(measurement);
106
118
  return measurement->object;
107
119
  }
108
120
 
109
- prof_measurement_t*
110
- prof_get_measurement(VALUE self)
121
+ prof_measurement_t* prof_get_measurement(VALUE self)
111
122
  {
112
123
  /* Can't use Data_Get_Struct because that triggers the event hook
113
124
  ending up in endless recursion. */
114
- prof_measurement_t* result = DATA_PTR(self);
125
+ prof_measurement_t* result = RTYPEDDATA_DATA(self);
115
126
 
116
127
  if (!result)
117
128
  rb_raise(rb_eRuntimeError, "This RubyProf::Measurement instance has already been freed, likely because its profile has been freed.");
@@ -123,8 +134,7 @@ prof_get_measurement(VALUE self)
123
134
  total_time -> float
124
135
 
125
136
  Returns the total amount of time spent in this method and its children. */
126
- static VALUE
127
- prof_measurement_total_time(VALUE self)
137
+ static VALUE prof_measurement_total_time(VALUE self)
128
138
  {
129
139
  prof_measurement_t* result = prof_get_measurement(self);
130
140
  return rb_float_new(result->total_time);
@@ -146,8 +156,7 @@ prof_measurement_self_time(VALUE self)
146
156
  wait_time -> float
147
157
 
148
158
  Returns the total amount of time this method waited for other threads. */
149
- static VALUE
150
- prof_measurement_wait_time(VALUE self)
159
+ static VALUE prof_measurement_wait_time(VALUE self)
151
160
  {
152
161
  prof_measurement_t* result = prof_get_measurement(self);
153
162
 
@@ -158,10 +167,9 @@ prof_measurement_wait_time(VALUE self)
158
167
  called -> int
159
168
 
160
169
  Returns the total amount of times this method was called. */
161
- static VALUE
162
- prof_measurement_called(VALUE self)
170
+ static VALUE prof_measurement_called(VALUE self)
163
171
  {
164
- prof_measurement_t *result = prof_get_measurement(self);
172
+ prof_measurement_t* result = prof_get_measurement(self);
165
173
  return INT2NUM(result->called);
166
174
  }
167
175
 
@@ -169,10 +177,9 @@ prof_measurement_called(VALUE self)
169
177
  called=n -> n
170
178
 
171
179
  Sets the call count to n. */
172
- static VALUE
173
- prof_measurement_set_called(VALUE self, VALUE called)
180
+ static VALUE prof_measurement_set_called(VALUE self, VALUE called)
174
181
  {
175
- prof_measurement_t *result = prof_get_measurement(self);
182
+ prof_measurement_t* result = prof_get_measurement(self);
176
183
  result->called = NUM2INT(called);
177
184
  return called;
178
185
  }
@@ -215,7 +222,7 @@ void rp_init_measure()
215
222
  rp_init_measure_allocations();
216
223
  rp_init_measure_memory();
217
224
 
218
- cRpMeasurement = rb_define_class_under(mProf, "Measurement", rb_cData);
225
+ cRpMeasurement = rb_define_class_under(mProf, "Measurement", rb_cObject);
219
226
  rb_undef_method(CLASS_OF(cRpMeasurement), "new");
220
227
  rb_define_alloc_func(cRpMeasurement, prof_measurement_allocate);
221
228
 
@@ -8,7 +8,7 @@
8
8
 
9
9
  extern VALUE mMeasure;
10
10
 
11
- typedef double (*get_measurement)(rb_trace_arg_t *trace_arg);
11
+ typedef double (*get_measurement)(rb_trace_arg_t* trace_arg);
12
12
 
13
13
  typedef enum
14
14
  {
@@ -18,7 +18,7 @@ typedef enum
18
18
  MEASURE_MEMORY
19
19
  } prof_measure_mode_t;
20
20
 
21
- typedef struct
21
+ typedef struct prof_measurer_t
22
22
  {
23
23
  get_measurement measure;
24
24
  prof_measure_mode_t mode;
@@ -36,14 +36,14 @@ typedef struct prof_measurement_t
36
36
  VALUE object;
37
37
  } prof_measurement_t;
38
38
 
39
- prof_measurer_t *prof_get_measurer(prof_measure_mode_t measure, bool track_allocations);
40
- double prof_measure(prof_measurer_t *measurer, rb_trace_arg_t* trace_arg);
39
+ prof_measurer_t* prof_get_measurer(prof_measure_mode_t measure, bool track_allocations);
40
+ double prof_measure(prof_measurer_t* measurer, rb_trace_arg_t* trace_arg);
41
41
 
42
- prof_measurement_t *prof_measurement_create(void);
42
+ prof_measurement_t* prof_measurement_create(void);
43
43
  void prof_measurement_free(prof_measurement_t* measurement);
44
- VALUE prof_measurement_wrap(prof_measurement_t *measurement);
44
+ VALUE prof_measurement_wrap(prof_measurement_t* measurement);
45
45
  prof_measurement_t* prof_get_measurement(VALUE self);
46
- void prof_measurement_mark(void *data);
46
+ void prof_measurement_mark(void* data);
47
47
 
48
48
  void rp_init_measure(void);
49
49
 
@@ -2,14 +2,13 @@
2
2
  Please see the LICENSE file for copyright and distribution information */
3
3
 
4
4
  #include "rp_allocation.h"
5
- #include "rp_call_info.h"
5
+ #include "rp_call_trees.h"
6
6
  #include "rp_method.h"
7
7
 
8
8
  VALUE cRpMethodInfo;
9
9
 
10
10
  /* ================ Helper Functions =================*/
11
- VALUE
12
- resolve_klass(VALUE klass, unsigned int *klass_flags)
11
+ VALUE resolve_klass(VALUE klass, unsigned int* klass_flags)
13
12
  {
14
13
  VALUE result = klass;
15
14
 
@@ -62,8 +61,7 @@ resolve_klass(VALUE klass, unsigned int *klass_flags)
62
61
  return result;
63
62
  }
64
63
 
65
- VALUE
66
- resolve_klass_name(VALUE klass, unsigned int* klass_flags)
64
+ VALUE resolve_klass_name(VALUE klass, unsigned int* klass_flags)
67
65
  {
68
66
  VALUE result = Qnil;
69
67
 
@@ -83,8 +81,7 @@ resolve_klass_name(VALUE klass, unsigned int* klass_flags)
83
81
  return result;
84
82
  }
85
83
 
86
- st_data_t
87
- method_key(VALUE klass, VALUE msym)
84
+ st_data_t method_key(VALUE klass, VALUE msym)
88
85
  {
89
86
  VALUE resolved_klass = klass;
90
87
 
@@ -104,21 +101,18 @@ method_key(VALUE klass, VALUE msym)
104
101
  }
105
102
 
106
103
  /* ====== Allocation Table ====== */
107
- st_table*
108
- allocations_table_create()
104
+ st_table* allocations_table_create()
109
105
  {
110
- return st_init_numtable();
106
+ return rb_st_init_numtable();
111
107
  }
112
108
 
113
- static int
114
- allocations_table_free_iterator(st_data_t key, st_data_t value, st_data_t dummy)
115
- {
109
+ static int allocations_table_free_iterator(st_data_t key, st_data_t value, st_data_t dummy)
110
+ {
116
111
  prof_allocation_free((prof_allocation_t*)value);
117
112
  return ST_CONTINUE;
118
113
  }
119
114
 
120
- static int
121
- prof_method_collect_allocations(st_data_t key, st_data_t value, st_data_t result)
115
+ static int prof_method_collect_allocations(st_data_t key, st_data_t value, st_data_t result)
122
116
  {
123
117
  prof_allocation_t* allocation = (prof_allocation_t*)value;
124
118
  VALUE arr = (VALUE)result;
@@ -126,28 +120,25 @@ prof_method_collect_allocations(st_data_t key, st_data_t value, st_data_t result
126
120
  return ST_CONTINUE;
127
121
  }
128
122
 
129
- static int
130
- prof_method_mark_allocations(st_data_t key, st_data_t value, st_data_t data)
123
+ static int prof_method_mark_allocations(st_data_t key, st_data_t value, st_data_t data)
131
124
  {
132
125
  prof_allocation_t* allocation = (prof_allocation_t*)value;
133
126
  prof_allocation_mark(allocation);
134
127
  return ST_CONTINUE;
135
128
  }
136
129
 
137
- void
138
- allocations_table_free(st_table* table)
130
+ void allocations_table_free(st_table* table)
139
131
  {
140
- st_foreach(table, allocations_table_free_iterator, 0);
141
- st_free_table(table);
132
+ rb_st_foreach(table, allocations_table_free_iterator, 0);
133
+ rb_st_free_table(table);
142
134
  }
143
135
 
144
136
  /* ================ prof_method_t =================*/
145
- static prof_method_t*
146
- prof_get_method(VALUE self)
137
+ prof_method_t* prof_get_method(VALUE self)
147
138
  {
148
139
  /* Can't use Data_Get_Struct because that triggers the event hook
149
140
  ending up in endless recursion. */
150
- prof_method_t* result = DATA_PTR(self);
141
+ prof_method_t* result = RTYPEDDATA_DATA(self);
151
142
 
152
143
  if (!result)
153
144
  rb_raise(rb_eRuntimeError, "This RubyProf::MethodInfo instance has already been freed, likely because its profile has been freed.");
@@ -155,27 +146,24 @@ prof_get_method(VALUE self)
155
146
  return result;
156
147
  }
157
148
 
158
- prof_method_t*
159
- prof_method_create(VALUE klass, VALUE msym, VALUE source_file, int source_line)
149
+ prof_method_t* prof_method_create(VALUE profile, VALUE klass, VALUE msym, VALUE source_file, int source_line)
160
150
  {
161
- prof_method_t *result = ALLOC(prof_method_t);
151
+ prof_method_t* result = ALLOC(prof_method_t);
152
+ result->profile = profile;
153
+
162
154
  result->key = method_key(klass, msym);
163
155
  result->klass_flags = 0;
164
156
 
165
- /* Note we do not call resolve_klass_name now because that causes an object allocation that shows up
157
+ /* Note we do not call resolve_klass_name now because that causes an object allocation that shows up
166
158
  in the allocation results so we want to avoid it until after the profile run is complete. */
167
159
  result->klass = resolve_klass(klass, &result->klass_flags);
168
160
  result->klass_name = Qnil;
169
161
  result->method_name = msym;
170
162
  result->measurement = prof_measurement_create();
171
163
 
172
- result->root = false;
173
- result->excluded = false;
174
-
175
- result->parent_call_infos = method_table_create();
176
- result->child_call_infos = method_table_create();
164
+ result->call_trees = prof_call_trees_create();
177
165
  result->allocations_table = allocations_table_create();
178
-
166
+
179
167
  result->visits = 0;
180
168
  result->recursive = false;
181
169
 
@@ -187,180 +175,127 @@ prof_method_create(VALUE klass, VALUE msym, VALUE source_file, int source_line)
187
175
  return result;
188
176
  }
189
177
 
190
- prof_method_t*
191
- prof_method_create_excluded(VALUE klass, VALUE msym)
192
- {
193
- prof_method_t* result = prof_method_create(klass, msym, Qnil, 0);
194
- result->excluded = 1;
195
- return result;
196
- }
197
-
198
- static int
199
- prof_method_collect_call_infos(st_data_t key, st_data_t value, st_data_t result)
200
- {
201
- prof_call_info_t* call_info = (prof_call_info_t*)value;
202
- VALUE arr = (VALUE)result;
203
- rb_ary_push(arr, prof_call_info_wrap(call_info));
204
- return ST_CONTINUE;
205
- }
206
-
207
- static int
208
- prof_method_mark_call_infos(st_data_t key, st_data_t value, st_data_t data)
209
- {
210
- prof_call_info_t* call_info = (prof_call_info_t*)value;
211
- prof_call_info_mark(call_info);
212
- return ST_CONTINUE;
213
- }
214
-
215
- static int
216
- call_infos_free_iterator(st_data_t key, st_data_t value, st_data_t dummy)
217
- {
218
- prof_call_info_free((prof_call_info_t*)value);
219
- return ST_CONTINUE;
220
- }
221
-
222
- void
223
- call_info_table_free(st_table* table)
224
- {
225
- st_foreach(table, call_infos_free_iterator, 0);
226
- st_free_table(table);
227
- }
228
-
229
178
  /* The underlying c structures are freed when the parent profile is freed.
230
179
  However, on shutdown the Ruby GC frees objects in any will-nilly order.
231
180
  That means the ruby thread object wrapping the c thread struct may
232
181
  be freed before the parent profile. Thus we add in a free function
233
182
  for the garbage collector so that if it does get called will nil
234
183
  out our Ruby object reference.*/
235
- static void
236
- prof_method_ruby_gc_free(void *data)
184
+ static void prof_method_ruby_gc_free(void* data)
237
185
  {
238
- prof_method_t* method = (prof_method_t*)data;
239
-
240
- /* Has this method object been accessed by Ruby? If
241
- yes clean it up so to avoid a segmentation fault. */
242
- if (method->object != Qnil)
243
- {
244
- RDATA(method->object)->dmark = NULL;
245
- RDATA(method->object)->dfree = NULL;
246
- RDATA(method->object)->data = NULL;
186
+ if (data)
187
+ {
188
+ prof_method_t* method = (prof_method_t*)data;
247
189
  method->object = Qnil;
248
190
  }
249
- method->klass_name = Qnil;
250
- method->method_name = Qnil;
251
191
  }
252
192
 
253
- static void
254
- prof_method_free(prof_method_t* method)
193
+ static void prof_method_free(prof_method_t* method)
255
194
  {
256
- prof_method_ruby_gc_free(method);
257
- allocations_table_free(method->allocations_table);
258
-
259
- /* Remember call infos are referenced by their parent method and child method, so we only want
260
- to iterate over one of them to avoid a double freeing */
261
- call_info_table_free(method->parent_call_infos);
262
- xfree(method->child_call_infos);
195
+ /* Has this method object been accessed by Ruby? If
196
+ yes clean it up so to avoid a segmentation fault. */
197
+ if (method->object != Qnil)
198
+ {
199
+ RTYPEDDATA(method->object)->data = NULL;
200
+ method->object = Qnil;
201
+ }
263
202
 
203
+ allocations_table_free(method->allocations_table);
204
+ prof_call_trees_free(method->call_trees);
264
205
  prof_measurement_free(method->measurement);
265
206
  xfree(method);
266
207
  }
267
208
 
268
- size_t
269
- prof_method_size(const void *data)
209
+ size_t prof_method_size(const void* data)
270
210
  {
271
211
  return sizeof(prof_method_t);
272
212
  }
273
213
 
274
- void
275
- prof_method_mark(void *data)
214
+ void prof_method_mark(void* data)
276
215
  {
216
+ if (!data) return;
217
+
277
218
  prof_method_t* method = (prof_method_t*)data;
219
+
220
+ if (method->profile != Qnil)
221
+ rb_gc_mark(method->profile);
222
+
223
+ if (method->object != Qnil)
224
+ rb_gc_mark(method->object);
225
+
278
226
  rb_gc_mark(method->klass_name);
279
227
  rb_gc_mark(method->method_name);
280
-
228
+ rb_gc_mark(method->source_file);
229
+
281
230
  if (method->klass != Qnil)
282
231
  rb_gc_mark(method->klass);
283
232
 
284
- if (method->object != Qnil)
285
- rb_gc_mark(method->object);
286
-
287
233
  prof_measurement_mark(method->measurement);
288
-
289
- st_foreach(method->parent_call_infos, prof_method_mark_call_infos, 0);
290
- st_foreach(method->child_call_infos, prof_method_mark_call_infos, 0);
291
- st_foreach(method->allocations_table, prof_method_mark_allocations, 0);
234
+
235
+ rb_st_foreach(method->allocations_table, prof_method_mark_allocations, 0);
292
236
  }
293
237
 
294
- static VALUE
295
- prof_method_allocate(VALUE klass)
238
+ static VALUE prof_method_allocate(VALUE klass)
296
239
  {
297
- prof_method_t* method_data = prof_method_create(Qnil, Qnil, Qnil, 0);
240
+ prof_method_t* method_data = prof_method_create(Qnil, Qnil, Qnil, Qnil, 0);
298
241
  method_data->object = prof_method_wrap(method_data);
299
242
  return method_data->object;
300
243
  }
301
244
 
302
- VALUE
303
- prof_method_wrap(prof_method_t *method)
245
+ static const rb_data_type_t method_info_type =
304
246
  {
305
- if (method->object == Qnil)
306
- {
307
- method->object = Data_Wrap_Struct(cRpMethodInfo, prof_method_mark, prof_method_ruby_gc_free, method);
308
- }
309
- return method->object;
310
- }
247
+ .wrap_struct_name = "MethodInfo",
248
+ .function =
249
+ {
250
+ .dmark = prof_method_mark,
251
+ .dfree = prof_method_ruby_gc_free,
252
+ .dsize = prof_method_size,
253
+ },
254
+ .data = NULL,
255
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY
256
+ };
311
257
 
312
- prof_method_t *
313
- prof_method_get(VALUE self)
258
+ VALUE prof_method_wrap(prof_method_t* method)
314
259
  {
315
- /* Can't use Data_Get_Struct because that triggers the event hook
316
- ending up in endless recursion. */
317
- prof_method_t* result = DATA_PTR(self);
318
-
319
- if (!result)
260
+ if (method->object == Qnil)
320
261
  {
321
- rb_raise(rb_eRuntimeError, "This RubyProf::MethodInfo instance has already been freed, likely because its profile has been freed.");
322
- }
323
-
324
- return result;
262
+ method->object = TypedData_Wrap_Struct(cRpMethodInfo, &method_info_type, method);
263
+ }
264
+ return method->object;
325
265
  }
326
266
 
327
- st_table *
328
- method_table_create()
267
+ st_table* method_table_create()
329
268
  {
330
- return st_init_numtable();
269
+ return rb_st_init_numtable();
331
270
  }
332
271
 
333
- static int
334
- method_table_free_iterator(st_data_t key, st_data_t value, st_data_t dummy)
272
+ static int method_table_free_iterator(st_data_t key, st_data_t value, st_data_t dummy)
335
273
  {
336
- prof_method_free((prof_method_t *)value);
274
+ prof_method_free((prof_method_t*)value);
337
275
  return ST_CONTINUE;
338
276
  }
339
277
 
340
- void
341
- method_table_free(st_table *table)
278
+ void method_table_free(st_table* table)
342
279
  {
343
- st_foreach(table, method_table_free_iterator, 0);
344
- st_free_table(table);
280
+ rb_st_foreach(table, method_table_free_iterator, 0);
281
+ rb_st_free_table(table);
345
282
  }
346
283
 
347
- size_t
348
- method_table_insert(st_table *table, st_data_t key, prof_method_t *val)
284
+ size_t method_table_insert(st_table* table, st_data_t key, prof_method_t* val)
349
285
  {
350
- return st_insert(table, (st_data_t) key, (st_data_t) val);
286
+ return rb_st_insert(table, (st_data_t)key, (st_data_t)val);
351
287
  }
352
288
 
353
- prof_method_t *
354
- method_table_lookup(st_table *table, st_data_t key)
289
+ prof_method_t* method_table_lookup(st_table* table, st_data_t key)
355
290
  {
356
291
  st_data_t val;
357
- if (st_lookup(table, (st_data_t)key, &val))
292
+ if (rb_st_lookup(table, (st_data_t)key, &val))
358
293
  {
359
- return (prof_method_t *) val;
294
+ return (prof_method_t*)val;
360
295
  }
361
296
  else
362
297
  {
363
- return NULL;
298
+ return NULL;
364
299
  }
365
300
  }
366
301
 
@@ -374,42 +309,15 @@ created. RubyProf::MethodInfo objects can be accessed via
374
309
  the RubyProf::Profile object.
375
310
  */
376
311
 
377
- /* call-seq:
378
- callers -> array
379
-
380
- Returns an array of call info objects that called this method (ie, parents).*/
381
- static VALUE
382
- prof_method_callers(VALUE self)
383
- {
384
- prof_method_t* method = prof_get_method(self);
385
- VALUE result = rb_ary_new();
386
- st_foreach(method->parent_call_infos, prof_method_collect_call_infos, result);
387
- return result;
388
- }
389
-
390
- /* call-seq:
391
- callees -> array
392
-
393
- Returns an array of call info objects that this method called (ie, children).*/
394
- static VALUE
395
- prof_method_callees(VALUE self)
396
- {
397
- prof_method_t* method = prof_get_method(self);
398
- VALUE result = rb_ary_new();
399
- st_foreach(method->child_call_infos, prof_method_collect_call_infos, result);
400
- return result;
401
- }
402
-
403
312
  /* call-seq:
404
313
  allocations -> array
405
314
 
406
315
  Returns an array of allocation information.*/
407
- static VALUE
408
- prof_method_allocations(VALUE self)
316
+ static VALUE prof_method_allocations(VALUE self)
409
317
  {
410
318
  prof_method_t* method = prof_get_method(self);
411
319
  VALUE result = rb_ary_new();
412
- st_foreach(method->allocations_table, prof_method_collect_allocations, result);
320
+ rb_st_foreach(method->allocations_table, prof_method_collect_allocations, result);
413
321
  return result;
414
322
  }
415
323
 
@@ -417,8 +325,7 @@ prof_method_allocations(VALUE self)
417
325
  called -> Measurement
418
326
 
419
327
  Returns the measurement associated with this method. */
420
- static VALUE
421
- prof_method_measurement(VALUE self)
328
+ static VALUE prof_method_measurement(VALUE self)
422
329
  {
423
330
  prof_method_t* method = prof_get_method(self);
424
331
  return prof_measurement_wrap(method->measurement);
@@ -431,7 +338,7 @@ return the source file of the method
431
338
  */
432
339
  static VALUE prof_method_source_file(VALUE self)
433
340
  {
434
- prof_method_t* method = prof_method_get(self);
341
+ prof_method_t* method = prof_get_method(self);
435
342
  return method->source_file;
436
343
  }
437
344
 
@@ -439,10 +346,9 @@ static VALUE prof_method_source_file(VALUE self)
439
346
  line_no -> int
440
347
 
441
348
  returns the line number of the method */
442
- static VALUE
443
- prof_method_line(VALUE self)
349
+ static VALUE prof_method_line(VALUE self)
444
350
  {
445
- prof_method_t* method = prof_method_get(self);
351
+ prof_method_t* method = prof_get_method(self);
446
352
  return INT2FIX(method->source_line);
447
353
  }
448
354
 
@@ -452,10 +358,9 @@ prof_method_line(VALUE self)
452
358
  Returns the name of this method's class. Singleton classes
453
359
  will have the form <Object::Object>. */
454
360
 
455
- static VALUE
456
- prof_method_klass_name(VALUE self)
361
+ static VALUE prof_method_klass_name(VALUE self)
457
362
  {
458
- prof_method_t *method = prof_method_get(self);
363
+ prof_method_t* method = prof_get_method(self);
459
364
  if (method->klass_name == Qnil)
460
365
  method->klass_name = resolve_klass_name(method->klass, &method->klass_flags);
461
366
 
@@ -467,10 +372,9 @@ prof_method_klass_name(VALUE self)
467
372
 
468
373
  Returns the klass flags */
469
374
 
470
- static VALUE
471
- prof_method_klass_flags(VALUE self)
375
+ static VALUE prof_method_klass_flags(VALUE self)
472
376
  {
473
- prof_method_t* method = prof_method_get(self);
377
+ prof_method_t* method = prof_get_method(self);
474
378
  return INT2FIX(method->klass_flags);
475
379
  }
476
380
 
@@ -480,51 +384,36 @@ prof_method_klass_flags(VALUE self)
480
384
  Returns the name of this method in the format Object#method. Singletons
481
385
  methods will be returned in the format <Object::Object>#method.*/
482
386
 
483
- static VALUE
484
- prof_method_name(VALUE self)
387
+ static VALUE prof_method_name(VALUE self)
485
388
  {
486
- prof_method_t *method = prof_method_get(self);
389
+ prof_method_t* method = prof_get_method(self);
487
390
  return method->method_name;
488
391
  }
489
392
 
490
- /* call-seq:
491
- root? -> boolean
492
-
493
- Returns the true if this method is at the top of the call stack */
494
- static VALUE
495
- prof_method_root(VALUE self)
496
- {
497
- prof_method_t *method = prof_method_get(self);
498
- return method->root ? Qtrue : Qfalse;
499
- }
500
-
501
393
  /* call-seq:
502
394
  recursive? -> boolean
503
395
 
504
396
  Returns the true if this method is recursively invoked */
505
- static VALUE
506
- prof_method_recursive(VALUE self)
397
+ static VALUE prof_method_recursive(VALUE self)
507
398
  {
508
- prof_method_t* method = prof_method_get(self);
399
+ prof_method_t* method = prof_get_method(self);
509
400
  return method->recursive ? Qtrue : Qfalse;
510
401
  }
511
402
 
512
403
  /* call-seq:
513
- excluded? -> boolean
404
+ call_trees -> CallTrees
514
405
 
515
- Returns the true if this method was excluded */
516
- static VALUE
517
- prof_method_excluded(VALUE self)
406
+ Returns the CallTrees associated with this method. */
407
+ static VALUE prof_method_call_trees(VALUE self)
518
408
  {
519
- prof_method_t* method = prof_method_get(self);
520
- return method->excluded ? Qtrue : Qfalse;
409
+ prof_method_t* method = prof_get_method(self);
410
+ return prof_call_trees_wrap(method->call_trees);
521
411
  }
522
412
 
523
413
  /* :nodoc: */
524
- static VALUE
525
- prof_method_dump(VALUE self)
414
+ static VALUE prof_method_dump(VALUE self)
526
415
  {
527
- prof_method_t* method_data = DATA_PTR(self);
416
+ prof_method_t* method_data = prof_get_method(self);
528
417
  VALUE result = rb_hash_new();
529
418
 
530
419
  rb_hash_aset(result, ID2SYM(rb_intern("klass_name")), prof_method_klass_name(self));
@@ -532,27 +421,21 @@ prof_method_dump(VALUE self)
532
421
  rb_hash_aset(result, ID2SYM(rb_intern("method_name")), method_data->method_name);
533
422
 
534
423
  rb_hash_aset(result, ID2SYM(rb_intern("key")), INT2FIX(method_data->key));
535
- rb_hash_aset(result, ID2SYM(rb_intern("root")), prof_method_root(self));
536
424
  rb_hash_aset(result, ID2SYM(rb_intern("recursive")), prof_method_recursive(self));
537
- rb_hash_aset(result, ID2SYM(rb_intern("excluded")), prof_method_excluded(self));
538
425
  rb_hash_aset(result, ID2SYM(rb_intern("source_file")), method_data->source_file);
539
426
  rb_hash_aset(result, ID2SYM(rb_intern("source_line")), INT2FIX(method_data->source_line));
540
427
 
428
+ rb_hash_aset(result, ID2SYM(rb_intern("call_trees")), prof_call_trees_wrap(method_data->call_trees));
541
429
  rb_hash_aset(result, ID2SYM(rb_intern("measurement")), prof_measurement_wrap(method_data->measurement));
542
-
543
- rb_hash_aset(result, ID2SYM(rb_intern("callers")), prof_method_callers(self));
544
- rb_hash_aset(result, ID2SYM(rb_intern("callees")), prof_method_callees(self));
545
-
546
430
  rb_hash_aset(result, ID2SYM(rb_intern("allocations")), prof_method_allocations(self));
547
431
 
548
432
  return result;
549
433
  }
550
434
 
551
435
  /* :nodoc: */
552
- static VALUE
553
- prof_method_load(VALUE self, VALUE data)
436
+ static VALUE prof_method_load(VALUE self, VALUE data)
554
437
  {
555
- prof_method_t* method_data = RDATA(self)->data;
438
+ prof_method_t* method_data = prof_get_method(self);
556
439
  method_data->object = self;
557
440
 
558
441
  method_data->klass_name = rb_hash_aref(data, ID2SYM(rb_intern("klass_name")));
@@ -560,42 +443,24 @@ prof_method_load(VALUE self, VALUE data)
560
443
  method_data->method_name = rb_hash_aref(data, ID2SYM(rb_intern("method_name")));
561
444
  method_data->key = FIX2LONG(rb_hash_aref(data, ID2SYM(rb_intern("key"))));
562
445
 
563
- method_data->root = rb_hash_aref(data, ID2SYM(rb_intern("root"))) == Qtrue ? true : false;
564
446
  method_data->recursive = rb_hash_aref(data, ID2SYM(rb_intern("recursive"))) == Qtrue ? true : false;
565
- method_data->excluded = rb_hash_aref(data, ID2SYM(rb_intern("excluded"))) == Qtrue ? true : false;
566
447
 
567
448
  method_data->source_file = rb_hash_aref(data, ID2SYM(rb_intern("source_file")));
568
449
  method_data->source_line = FIX2INT(rb_hash_aref(data, ID2SYM(rb_intern("source_line"))));
569
450
 
451
+ VALUE call_trees = rb_hash_aref(data, ID2SYM(rb_intern("call_trees")));
452
+ method_data->call_trees = prof_get_call_trees(call_trees);
453
+
570
454
  VALUE measurement = rb_hash_aref(data, ID2SYM(rb_intern("measurement")));
571
455
  method_data->measurement = prof_get_measurement(measurement);
572
456
 
573
- VALUE callers = rb_hash_aref(data, ID2SYM(rb_intern("callers")));
574
- for (int i = 0; i < rb_array_len(callers); i++)
575
- {
576
- VALUE call_info = rb_ary_entry(callers, i);
577
- prof_call_info_t *call_info_data = prof_get_call_info(call_info);
578
- st_data_t key = call_info_data->parent ? call_info_data->parent->key : method_key(Qnil, 0);
579
- call_info_table_insert(method_data->parent_call_infos, key, call_info_data);
580
- }
581
-
582
- VALUE callees = rb_hash_aref(data, ID2SYM(rb_intern("callees")));
583
- for (int i = 0; i < rb_array_len(callees); i++)
584
- {
585
- VALUE call_info = rb_ary_entry(callees, i);
586
- prof_call_info_t *call_info_data = prof_get_call_info(call_info);
587
-
588
- st_data_t key = call_info_data->method ? call_info_data->method->key : method_key(Qnil, 0);
589
- call_info_table_insert(method_data->child_call_infos, key, call_info_data);
590
- }
591
-
592
457
  VALUE allocations = rb_hash_aref(data, ID2SYM(rb_intern("allocations")));
593
458
  for (int i = 0; i < rb_array_len(allocations); i++)
594
459
  {
595
460
  VALUE allocation = rb_ary_entry(allocations, i);
596
461
  prof_allocation_t* allocation_data = prof_allocation_get(allocation);
597
462
 
598
- st_insert(method_data->allocations_table, allocation_data->key, (st_data_t)allocation_data);
463
+ rb_st_insert(method_data->allocations_table, allocation_data->key, (st_data_t)allocation_data);
599
464
  }
600
465
  return data;
601
466
  }
@@ -603,28 +468,24 @@ prof_method_load(VALUE self, VALUE data)
603
468
  void rp_init_method_info()
604
469
  {
605
470
  /* MethodInfo */
606
- cRpMethodInfo = rb_define_class_under(mProf, "MethodInfo", rb_cData);
471
+ cRpMethodInfo = rb_define_class_under(mProf, "MethodInfo", rb_cObject);
607
472
  rb_undef_method(CLASS_OF(cRpMethodInfo), "new");
608
473
  rb_define_alloc_func(cRpMethodInfo, prof_method_allocate);
609
474
 
610
475
  rb_define_method(cRpMethodInfo, "klass_name", prof_method_klass_name, 0);
611
476
  rb_define_method(cRpMethodInfo, "klass_flags", prof_method_klass_flags, 0);
612
-
613
477
  rb_define_method(cRpMethodInfo, "method_name", prof_method_name, 0);
614
-
615
- rb_define_method(cRpMethodInfo, "callers", prof_method_callers, 0);
616
- rb_define_method(cRpMethodInfo, "callees", prof_method_callees, 0);
617
- rb_define_method(cRpMethodInfo, "allocations", prof_method_allocations, 0);
618
478
 
479
+ rb_define_method(cRpMethodInfo, "call_trees", prof_method_call_trees, 0);
480
+
481
+ rb_define_method(cRpMethodInfo, "allocations", prof_method_allocations, 0);
619
482
  rb_define_method(cRpMethodInfo, "measurement", prof_method_measurement, 0);
620
-
483
+
621
484
  rb_define_method(cRpMethodInfo, "source_file", prof_method_source_file, 0);
622
485
  rb_define_method(cRpMethodInfo, "line", prof_method_line, 0);
623
486
 
624
- rb_define_method(cRpMethodInfo, "root?", prof_method_root, 0);
625
487
  rb_define_method(cRpMethodInfo, "recursive?", prof_method_recursive, 0);
626
- rb_define_method(cRpMethodInfo, "excluded?", prof_method_excluded, 0);
627
488
 
628
489
  rb_define_method(cRpMethodInfo, "_dump_data", prof_method_dump, 0);
629
490
  rb_define_method(cRpMethodInfo, "_load_data", prof_method_load, 1);
630
- }
491
+ }