ruby-prof 1.5.0-x64-mingw-ucrt → 1.6.2-x64-mingw-ucrt
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.
- checksums.yaml +4 -4
- data/CHANGES +19 -0
- data/bin/ruby-prof +105 -87
- data/ext/ruby_prof/rp_allocation.c +136 -81
- data/ext/ruby_prof/rp_allocation.h +8 -6
- data/ext/ruby_prof/rp_call_tree.c +502 -457
- data/ext/ruby_prof/rp_call_tree.h +47 -44
- data/ext/ruby_prof/rp_call_trees.c +1 -1
- data/ext/ruby_prof/rp_measurement.c +10 -3
- data/ext/ruby_prof/rp_method.c +86 -79
- data/ext/ruby_prof/rp_method.h +63 -62
- data/ext/ruby_prof/rp_profile.c +933 -948
- data/ext/ruby_prof/rp_profile.h +1 -0
- data/ext/ruby_prof/rp_thread.c +433 -410
- data/ext/ruby_prof/rp_thread.h +39 -39
- data/ext/ruby_prof/vc/ruby_prof.vcxproj +6 -3
- data/lib/3.1/ruby_prof.so +0 -0
- data/lib/3.2/ruby_prof.so +0 -0
- data/lib/ruby-prof/compatibility.rb +14 -0
- data/lib/ruby-prof/printers/abstract_printer.rb +2 -1
- data/lib/ruby-prof/printers/call_tree_printer.rb +1 -1
- data/lib/ruby-prof/printers/multi_printer.rb +17 -17
- data/lib/ruby-prof/profile.rb +70 -70
- data/lib/ruby-prof/rack.rb +31 -21
- data/lib/ruby-prof/version.rb +1 -1
- data/test/abstract_printer_test.rb +1 -0
- data/test/alias_test.rb +6 -11
- data/test/call_tree_test.rb +94 -197
- data/test/call_tree_visitor_test.rb +1 -6
- data/test/call_trees_test.rb +2 -2
- data/test/{basic_test.rb → compatibility_test.rb} +8 -2
- data/test/duplicate_names_test.rb +1 -1
- data/test/dynamic_method_test.rb +1 -6
- data/test/enumerable_test.rb +1 -1
- data/test/exceptions_test.rb +2 -2
- data/test/exclude_methods_test.rb +3 -8
- data/test/exclude_threads_test.rb +4 -9
- data/test/fiber_test.rb +2 -58
- data/test/gc_test.rb +2 -2
- data/test/inverse_call_tree_test.rb +33 -34
- data/test/line_number_test.rb +1 -1
- data/test/marshal_test.rb +3 -3
- data/test/measure_allocations_test.rb +8 -17
- data/test/measure_memory_test.rb +3 -12
- data/test/measure_process_time_test.rb +32 -36
- data/test/measure_wall_time_test.rb +176 -181
- data/test/merge_test.rb +146 -0
- data/test/multi_printer_test.rb +0 -5
- data/test/no_method_class_test.rb +1 -1
- data/test/pause_resume_test.rb +12 -16
- data/test/printer_call_stack_test.rb +2 -2
- data/test/printer_call_tree_test.rb +2 -2
- data/test/printer_flat_test.rb +1 -1
- data/test/printer_graph_html_test.rb +2 -2
- data/test/printer_graph_test.rb +2 -2
- data/test/printers_test.rb +14 -20
- data/test/printing_recursive_graph_test.rb +2 -2
- data/test/recursive_test.rb +2 -7
- data/test/scheduler.rb +9 -0
- data/test/singleton_test.rb +1 -1
- data/test/stack_printer_test.rb +5 -8
- data/test/start_stop_test.rb +11 -14
- data/test/test_helper.rb +7 -0
- data/test/thread_test.rb +84 -19
- data/test/unique_call_path_test.rb +4 -4
- data/test/yarv_test.rb +3 -3
- metadata +6 -5
@@ -1,44 +1,47 @@
|
|
1
|
-
/* Copyright (C) 2005-2019 Shugo Maeda <shugo@ruby-lang.org> and Charlie Savage <cfis@savagexi.com>
|
2
|
-
Please see the LICENSE file for copyright and distribution information */
|
3
|
-
|
4
|
-
#ifndef __RP_CALL_TREE_H__
|
5
|
-
#define __RP_CALL_TREE_H__
|
6
|
-
|
7
|
-
#include "ruby_prof.h"
|
8
|
-
#include "rp_measurement.h"
|
9
|
-
#include "rp_method.h"
|
10
|
-
|
11
|
-
extern VALUE cRpCallTree;
|
12
|
-
|
13
|
-
/* Callers and callee information for a method. */
|
14
|
-
typedef struct prof_call_tree_t
|
15
|
-
{
|
16
|
-
prof_owner_t owner;
|
17
|
-
prof_method_t* method;
|
18
|
-
struct prof_call_tree_t* parent;
|
19
|
-
st_table* children; /* Call infos that this call info calls */
|
20
|
-
prof_measurement_t* measurement;
|
21
|
-
VALUE object;
|
22
|
-
|
23
|
-
int visits; /* Current visits on the stack */
|
24
|
-
|
25
|
-
unsigned int source_line;
|
26
|
-
VALUE source_file;
|
27
|
-
} prof_call_tree_t;
|
28
|
-
|
29
|
-
prof_call_tree_t* prof_call_tree_create(prof_method_t* method, prof_call_tree_t* parent, VALUE source_file, int source_line);
|
30
|
-
prof_call_tree_t* prof_call_tree_copy(prof_call_tree_t* other);
|
31
|
-
void prof_call_tree_merge_internal(prof_call_tree_t* destination, prof_call_tree_t* other);
|
32
|
-
void prof_call_tree_mark(void* data);
|
33
|
-
prof_call_tree_t* call_tree_table_lookup(st_table* table, st_data_t key);
|
34
|
-
|
35
|
-
void prof_call_tree_add_parent(prof_call_tree_t* self, prof_call_tree_t* parent);
|
36
|
-
void prof_call_tree_add_child(prof_call_tree_t* self, prof_call_tree_t* child);
|
37
|
-
|
38
|
-
uint32_t
|
39
|
-
prof_call_tree_t*
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
1
|
+
/* Copyright (C) 2005-2019 Shugo Maeda <shugo@ruby-lang.org> and Charlie Savage <cfis@savagexi.com>
|
2
|
+
Please see the LICENSE file for copyright and distribution information */
|
3
|
+
|
4
|
+
#ifndef __RP_CALL_TREE_H__
|
5
|
+
#define __RP_CALL_TREE_H__
|
6
|
+
|
7
|
+
#include "ruby_prof.h"
|
8
|
+
#include "rp_measurement.h"
|
9
|
+
#include "rp_method.h"
|
10
|
+
|
11
|
+
extern VALUE cRpCallTree;
|
12
|
+
|
13
|
+
/* Callers and callee information for a method. */
|
14
|
+
typedef struct prof_call_tree_t
|
15
|
+
{
|
16
|
+
prof_owner_t owner;
|
17
|
+
prof_method_t* method;
|
18
|
+
struct prof_call_tree_t* parent;
|
19
|
+
st_table* children; /* Call infos that this call info calls */
|
20
|
+
prof_measurement_t* measurement;
|
21
|
+
VALUE object;
|
22
|
+
|
23
|
+
int visits; /* Current visits on the stack */
|
24
|
+
|
25
|
+
unsigned int source_line;
|
26
|
+
VALUE source_file;
|
27
|
+
} prof_call_tree_t;
|
28
|
+
|
29
|
+
prof_call_tree_t* prof_call_tree_create(prof_method_t* method, prof_call_tree_t* parent, VALUE source_file, int source_line);
|
30
|
+
prof_call_tree_t* prof_call_tree_copy(prof_call_tree_t* other);
|
31
|
+
void prof_call_tree_merge_internal(prof_call_tree_t* destination, prof_call_tree_t* other, st_table* method_table);
|
32
|
+
void prof_call_tree_mark(void* data);
|
33
|
+
prof_call_tree_t* call_tree_table_lookup(st_table* table, st_data_t key);
|
34
|
+
|
35
|
+
void prof_call_tree_add_parent(prof_call_tree_t* self, prof_call_tree_t* parent);
|
36
|
+
void prof_call_tree_add_child(prof_call_tree_t* self, prof_call_tree_t* child);
|
37
|
+
|
38
|
+
uint32_t prof_call_tree_figure_depth(prof_call_tree_t* call_tree);
|
39
|
+
VALUE prof_call_tree_methods(prof_call_tree_t* call_tree);
|
40
|
+
|
41
|
+
prof_call_tree_t* prof_get_call_tree(VALUE self);
|
42
|
+
VALUE prof_call_tree_wrap(prof_call_tree_t* call_tree);
|
43
|
+
void prof_call_tree_free(prof_call_tree_t* call_tree);
|
44
|
+
|
45
|
+
void rp_init_call_tree();
|
46
|
+
|
47
|
+
#endif //__RP_CALL_TREE_H__
|
@@ -170,7 +170,7 @@ VALUE prof_call_trees_min_depth(VALUE self)
|
|
170
170
|
prof_call_trees_t* call_trees = prof_get_call_trees(self);
|
171
171
|
for (prof_call_tree_t** p_call_tree = call_trees->start; p_call_tree < call_trees->ptr; p_call_tree++)
|
172
172
|
{
|
173
|
-
unsigned int call_tree_depth =
|
173
|
+
unsigned int call_tree_depth = prof_call_tree_figure_depth(*p_call_tree);
|
174
174
|
if (call_tree_depth < depth)
|
175
175
|
depth = call_tree_depth;
|
176
176
|
}
|
@@ -102,10 +102,16 @@ void prof_measurement_mark(void* data)
|
|
102
102
|
{
|
103
103
|
if (!data) return;
|
104
104
|
|
105
|
-
prof_measurement_t*
|
105
|
+
prof_measurement_t* measurement = (prof_measurement_t*)data;
|
106
106
|
|
107
|
-
if (
|
108
|
-
|
107
|
+
if (measurement->object != Qnil)
|
108
|
+
rb_gc_mark_movable(measurement->object);
|
109
|
+
}
|
110
|
+
|
111
|
+
void prof_measurement_compact(void* data)
|
112
|
+
{
|
113
|
+
prof_measurement_t* measurement = (prof_measurement_t*)data;
|
114
|
+
measurement->object = rb_gc_location(measurement->object);
|
109
115
|
}
|
110
116
|
|
111
117
|
void prof_measurement_free(prof_measurement_t* measurement)
|
@@ -155,6 +161,7 @@ static const rb_data_type_t measurement_type =
|
|
155
161
|
.dmark = prof_measurement_mark,
|
156
162
|
.dfree = prof_measurement_ruby_gc_free,
|
157
163
|
.dsize = prof_measurement_size,
|
164
|
+
.dcompact = prof_measurement_compact
|
158
165
|
},
|
159
166
|
.data = NULL,
|
160
167
|
.flags = RUBY_TYPED_FREE_IMMEDIATELY
|
data/ext/ruby_prof/rp_method.c
CHANGED
@@ -22,31 +22,38 @@ VALUE resolve_klass(VALUE klass, unsigned int* klass_flags)
|
|
22
22
|
figure out what it is attached to.*/
|
23
23
|
VALUE attached = rb_iv_get(klass, "__attached__");
|
24
24
|
|
25
|
-
|
26
|
-
if (BUILTIN_TYPE(attached) == T_CLASS)
|
25
|
+
switch (BUILTIN_TYPE(attached))
|
27
26
|
{
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
27
|
+
/* Is this a singleton class acting as a metaclass? */
|
28
|
+
case T_CLASS:
|
29
|
+
{
|
30
|
+
*klass_flags |= kClassSingleton;
|
31
|
+
result = attached;
|
32
|
+
break;
|
33
|
+
}
|
34
|
+
/* Is this for singleton methods on a module? */
|
35
|
+
case T_MODULE:
|
36
|
+
{
|
37
|
+
*klass_flags |= kModuleSingleton;
|
38
|
+
result = attached;
|
39
|
+
break;
|
40
|
+
}
|
41
|
+
/* Is this for singleton methods on an object? */
|
42
|
+
case T_OBJECT:
|
43
|
+
{
|
44
|
+
*klass_flags |= kObjectSingleton;
|
45
|
+
result = rb_class_superclass(klass);
|
46
|
+
break;
|
47
|
+
}
|
48
|
+
/* Ok, this could be other things like an array put onto
|
49
|
+
a singleton object (yeah, it happens, see the singleton
|
50
|
+
objects test case). */
|
51
|
+
default:
|
52
|
+
{
|
53
|
+
*klass_flags |= kOtherSingleton;
|
54
|
+
result = klass;
|
55
|
+
break;
|
56
|
+
}
|
50
57
|
}
|
51
58
|
}
|
52
59
|
/* Is this an include for a module? If so get the actual
|
@@ -56,7 +63,7 @@ VALUE resolve_klass(VALUE klass, unsigned int* klass_flags)
|
|
56
63
|
{
|
57
64
|
unsigned int dummy;
|
58
65
|
*klass_flags |= kModuleIncludee;
|
59
|
-
result = resolve_klass(
|
66
|
+
result = resolve_klass(RBASIC_CLASS(klass), &dummy);
|
60
67
|
}
|
61
68
|
return result;
|
62
69
|
}
|
@@ -94,7 +101,7 @@ st_data_t method_key(VALUE klass, VALUE msym)
|
|
94
101
|
}
|
95
102
|
else if (BUILTIN_TYPE(klass) == T_ICLASS)
|
96
103
|
{
|
97
|
-
resolved_klass =
|
104
|
+
resolved_klass = RBASIC_CLASS(klass);
|
98
105
|
}
|
99
106
|
|
100
107
|
st_data_t hash = rb_hash_start(0);
|
@@ -105,39 +112,6 @@ st_data_t method_key(VALUE klass, VALUE msym)
|
|
105
112
|
return hash;
|
106
113
|
}
|
107
114
|
|
108
|
-
/* ====== Allocation Table ====== */
|
109
|
-
st_table* allocations_table_create()
|
110
|
-
{
|
111
|
-
return rb_st_init_numtable();
|
112
|
-
}
|
113
|
-
|
114
|
-
static int allocations_table_free_iterator(st_data_t key, st_data_t value, st_data_t dummy)
|
115
|
-
{
|
116
|
-
prof_allocation_free((prof_allocation_t*)value);
|
117
|
-
return ST_CONTINUE;
|
118
|
-
}
|
119
|
-
|
120
|
-
static int prof_method_collect_allocations(st_data_t key, st_data_t value, st_data_t result)
|
121
|
-
{
|
122
|
-
prof_allocation_t* allocation = (prof_allocation_t*)value;
|
123
|
-
VALUE arr = (VALUE)result;
|
124
|
-
rb_ary_push(arr, prof_allocation_wrap(allocation));
|
125
|
-
return ST_CONTINUE;
|
126
|
-
}
|
127
|
-
|
128
|
-
static int prof_method_mark_allocations(st_data_t key, st_data_t value, st_data_t data)
|
129
|
-
{
|
130
|
-
prof_allocation_t* allocation = (prof_allocation_t*)value;
|
131
|
-
prof_allocation_mark(allocation);
|
132
|
-
return ST_CONTINUE;
|
133
|
-
}
|
134
|
-
|
135
|
-
void allocations_table_free(st_table* table)
|
136
|
-
{
|
137
|
-
rb_st_foreach(table, allocations_table_free_iterator, 0);
|
138
|
-
rb_st_free_table(table);
|
139
|
-
}
|
140
|
-
|
141
115
|
/* ================ prof_method_t =================*/
|
142
116
|
prof_method_t* prof_get_method(VALUE self)
|
143
117
|
{
|
@@ -167,7 +141,7 @@ prof_method_t* prof_method_create(VALUE profile, VALUE klass, VALUE msym, VALUE
|
|
167
141
|
result->measurement = prof_measurement_create();
|
168
142
|
|
169
143
|
result->call_trees = prof_call_trees_create();
|
170
|
-
result->allocations_table =
|
144
|
+
result->allocations_table = prof_allocations_create();
|
171
145
|
|
172
146
|
result->visits = 0;
|
173
147
|
result->recursive = false;
|
@@ -180,6 +154,14 @@ prof_method_t* prof_method_create(VALUE profile, VALUE klass, VALUE msym, VALUE
|
|
180
154
|
return result;
|
181
155
|
}
|
182
156
|
|
157
|
+
prof_method_t* prof_method_copy(prof_method_t* other)
|
158
|
+
{
|
159
|
+
prof_method_t* result = prof_method_create(other->profile, other->klass, other->method_name, other->source_file, other->source_line);
|
160
|
+
result->measurement = prof_measurement_copy(other->measurement);
|
161
|
+
|
162
|
+
return result;
|
163
|
+
}
|
164
|
+
|
183
165
|
/* The underlying c structures are freed when the parent profile is freed.
|
184
166
|
However, on shutdown the Ruby GC frees objects in any will-nilly order.
|
185
167
|
That means the ruby thread object wrapping the c thread struct may
|
@@ -205,7 +187,7 @@ static void prof_method_free(prof_method_t* method)
|
|
205
187
|
method->object = Qnil;
|
206
188
|
}
|
207
189
|
|
208
|
-
|
190
|
+
prof_allocations_free(method->allocations_table);
|
209
191
|
prof_call_trees_free(method->call_trees);
|
210
192
|
prof_measurement_free(method->measurement);
|
211
193
|
xfree(method);
|
@@ -223,21 +205,29 @@ void prof_method_mark(void* data)
|
|
223
205
|
prof_method_t* method = (prof_method_t*)data;
|
224
206
|
|
225
207
|
if (method->profile != Qnil)
|
226
|
-
|
208
|
+
rb_gc_mark_movable(method->profile);
|
227
209
|
|
228
210
|
if (method->object != Qnil)
|
229
|
-
|
211
|
+
rb_gc_mark_movable(method->object);
|
230
212
|
|
231
|
-
|
232
|
-
|
213
|
+
rb_gc_mark_movable(method->klass_name);
|
214
|
+
rb_gc_mark_movable(method->method_name);
|
233
215
|
rb_gc_mark(method->source_file);
|
234
216
|
|
235
217
|
if (method->klass != Qnil)
|
236
218
|
rb_gc_mark(method->klass);
|
237
219
|
|
238
220
|
prof_measurement_mark(method->measurement);
|
221
|
+
prof_allocations_mark(method->allocations_table);
|
222
|
+
}
|
239
223
|
|
240
|
-
|
224
|
+
void prof_method_compact(void* data)
|
225
|
+
{
|
226
|
+
prof_method_t* method = (prof_method_t*)data;
|
227
|
+
method->object = rb_gc_location(method->object);
|
228
|
+
method->profile = rb_gc_location(method->profile);
|
229
|
+
method->klass_name = rb_gc_location(method->klass_name);
|
230
|
+
method->method_name = rb_gc_location(method->method_name);
|
241
231
|
}
|
242
232
|
|
243
233
|
static VALUE prof_method_allocate(VALUE klass)
|
@@ -255,6 +245,7 @@ static const rb_data_type_t method_info_type =
|
|
255
245
|
.dmark = prof_method_mark,
|
256
246
|
.dfree = prof_method_ruby_gc_free,
|
257
247
|
.dsize = prof_method_size,
|
248
|
+
.dcompact = prof_method_compact
|
258
249
|
},
|
259
250
|
.data = NULL,
|
260
251
|
.flags = RUBY_TYPED_FREE_IMMEDIATELY
|
@@ -291,6 +282,30 @@ size_t method_table_insert(st_table* table, st_data_t key, prof_method_t* val)
|
|
291
282
|
return rb_st_insert(table, (st_data_t)key, (st_data_t)val);
|
292
283
|
}
|
293
284
|
|
285
|
+
static int prof_method_table_merge_internal(st_data_t key, st_data_t value, st_data_t data)
|
286
|
+
{
|
287
|
+
st_table* self_table = (st_table*)data;
|
288
|
+
prof_method_t* other_child = (prof_method_t*)value;
|
289
|
+
|
290
|
+
prof_method_t* self_child = method_table_lookup(self_table, other_child->key);
|
291
|
+
if (self_child)
|
292
|
+
{
|
293
|
+
prof_measurement_merge_internal(self_child->measurement, other_child->measurement);
|
294
|
+
}
|
295
|
+
else
|
296
|
+
{
|
297
|
+
prof_method_t* copy = prof_method_copy(other_child);
|
298
|
+
method_table_insert(self_table, copy->key, copy);
|
299
|
+
}
|
300
|
+
|
301
|
+
return ST_CONTINUE;
|
302
|
+
}
|
303
|
+
|
304
|
+
void prof_method_table_merge(st_table* self, st_table* other)
|
305
|
+
{
|
306
|
+
rb_st_foreach(other, prof_method_table_merge_internal, (st_data_t)self);
|
307
|
+
}
|
308
|
+
|
294
309
|
prof_method_t* method_table_lookup(st_table* table, st_data_t key)
|
295
310
|
{
|
296
311
|
st_data_t val;
|
@@ -360,9 +375,7 @@ Returns an array of allocation information.*/
|
|
360
375
|
static VALUE prof_method_allocations(VALUE self)
|
361
376
|
{
|
362
377
|
prof_method_t* method = prof_get_method(self);
|
363
|
-
|
364
|
-
rb_st_foreach(method->allocations_table, prof_method_collect_allocations, result);
|
365
|
-
return result;
|
378
|
+
return prof_allocations_wrap(method->allocations_table);
|
366
379
|
}
|
367
380
|
|
368
381
|
/* call-seq:
|
@@ -397,7 +410,7 @@ static VALUE prof_method_line(VALUE self)
|
|
397
410
|
}
|
398
411
|
|
399
412
|
/* call-seq:
|
400
|
-
klass_name ->
|
413
|
+
klass_name -> String
|
401
414
|
|
402
415
|
Returns the name of this method's class. Singleton classes
|
403
416
|
will have the form <Object::Object>. */
|
@@ -423,9 +436,9 @@ static VALUE prof_method_klass_flags(VALUE self)
|
|
423
436
|
}
|
424
437
|
|
425
438
|
/* call-seq:
|
426
|
-
method_name ->
|
439
|
+
method_name -> Symbol
|
427
440
|
|
428
|
-
Returns the name of this method in the format Object#method.
|
441
|
+
Returns the name of this method in the format Object#method. Singletons
|
429
442
|
methods will be returned in the format <Object::Object>#method.*/
|
430
443
|
|
431
444
|
static VALUE prof_method_name(VALUE self)
|
@@ -499,13 +512,7 @@ static VALUE prof_method_load(VALUE self, VALUE data)
|
|
499
512
|
method_data->measurement = prof_get_measurement(measurement);
|
500
513
|
|
501
514
|
VALUE allocations = rb_hash_aref(data, ID2SYM(rb_intern("allocations")));
|
502
|
-
|
503
|
-
{
|
504
|
-
VALUE allocation = rb_ary_entry(allocations, i);
|
505
|
-
prof_allocation_t* allocation_data = prof_allocation_get(allocation);
|
506
|
-
|
507
|
-
rb_st_insert(method_data->allocations_table, allocation_data->key, (st_data_t)allocation_data);
|
508
|
-
}
|
515
|
+
prof_allocations_unwrap(method_data->allocations_table, allocations);
|
509
516
|
return data;
|
510
517
|
}
|
511
518
|
|
data/ext/ruby_prof/rp_method.h
CHANGED
@@ -1,62 +1,63 @@
|
|
1
|
-
/* Copyright (C) 2005-2019 Shugo Maeda <shugo@ruby-lang.org> and Charlie Savage <cfis@savagexi.com>
|
2
|
-
Please see the LICENSE file for copyright and distribution information */
|
3
|
-
|
4
|
-
#ifndef __RP_METHOD_INFO__
|
5
|
-
#define __RP_METHOD_INFO__
|
6
|
-
|
7
|
-
#include "ruby_prof.h"
|
8
|
-
#include "rp_measurement.h"
|
9
|
-
|
10
|
-
extern VALUE cRpMethodInfo;
|
11
|
-
|
12
|
-
// Source relation bit offsets.
|
13
|
-
enum {
|
14
|
-
kModuleIncludee = 0x1, // Included in module
|
15
|
-
kClassSingleton = 0x2, // Singleton of a class
|
16
|
-
kModuleSingleton = 0x4, // Singleton of a module
|
17
|
-
kObjectSingleton = 0x8, // Singleton of an object
|
18
|
-
kOtherSingleton = 0x10 // Singleton of unknown object
|
19
|
-
};
|
20
|
-
|
21
|
-
// Profiling information for each method.
|
22
|
-
// Excluded methods have no call_trees, source_klass, or source_file.
|
23
|
-
typedef struct prof_method_t
|
24
|
-
{
|
25
|
-
VALUE profile; // Profile this method is associated with - needed for mark phase
|
26
|
-
struct prof_call_trees_t* call_trees; // Call infos that call this method
|
27
|
-
st_table* allocations_table; // Tracks object allocations
|
28
|
-
|
29
|
-
st_data_t key; // Table key
|
30
|
-
unsigned int klass_flags; // Information about the type of class
|
31
|
-
VALUE klass; // Resolved klass
|
32
|
-
VALUE klass_name; // Resolved klass name for this method
|
33
|
-
VALUE method_name; // Resolved method name for this method
|
34
|
-
|
35
|
-
VALUE object; // Cached ruby object
|
36
|
-
|
37
|
-
bool recursive;
|
38
|
-
int visits; // Current visits on the stack
|
39
|
-
VALUE source_file; // Source file
|
40
|
-
int source_line; // Line number
|
41
|
-
|
42
|
-
prof_measurement_t* measurement; // Stores measurement data for this method
|
43
|
-
} prof_method_t;
|
44
|
-
|
45
|
-
void rp_init_method_info(void);
|
46
|
-
|
47
|
-
st_data_t method_key(VALUE klass, VALUE msym);
|
48
|
-
|
49
|
-
st_table* method_table_create(void);
|
50
|
-
prof_method_t* method_table_lookup(st_table* table, st_data_t key);
|
51
|
-
size_t method_table_insert(st_table* table, st_data_t key, prof_method_t* val);
|
52
|
-
void method_table_free(st_table* table);
|
53
|
-
|
54
|
-
prof_method_t*
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
VALUE
|
61
|
-
|
62
|
-
|
1
|
+
/* Copyright (C) 2005-2019 Shugo Maeda <shugo@ruby-lang.org> and Charlie Savage <cfis@savagexi.com>
|
2
|
+
Please see the LICENSE file for copyright and distribution information */
|
3
|
+
|
4
|
+
#ifndef __RP_METHOD_INFO__
|
5
|
+
#define __RP_METHOD_INFO__
|
6
|
+
|
7
|
+
#include "ruby_prof.h"
|
8
|
+
#include "rp_measurement.h"
|
9
|
+
|
10
|
+
extern VALUE cRpMethodInfo;
|
11
|
+
|
12
|
+
// Source relation bit offsets.
|
13
|
+
enum {
|
14
|
+
kModuleIncludee = 0x1, // Included in module
|
15
|
+
kClassSingleton = 0x2, // Singleton of a class
|
16
|
+
kModuleSingleton = 0x4, // Singleton of a module
|
17
|
+
kObjectSingleton = 0x8, // Singleton of an object
|
18
|
+
kOtherSingleton = 0x10 // Singleton of unknown object
|
19
|
+
};
|
20
|
+
|
21
|
+
// Profiling information for each method.
|
22
|
+
// Excluded methods have no call_trees, source_klass, or source_file.
|
23
|
+
typedef struct prof_method_t
|
24
|
+
{
|
25
|
+
VALUE profile; // Profile this method is associated with - needed for mark phase
|
26
|
+
struct prof_call_trees_t* call_trees; // Call infos that call this method
|
27
|
+
st_table* allocations_table; // Tracks object allocations
|
28
|
+
|
29
|
+
st_data_t key; // Table key
|
30
|
+
unsigned int klass_flags; // Information about the type of class
|
31
|
+
VALUE klass; // Resolved klass
|
32
|
+
VALUE klass_name; // Resolved klass name for this method
|
33
|
+
VALUE method_name; // Resolved method name for this method
|
34
|
+
|
35
|
+
VALUE object; // Cached ruby object
|
36
|
+
|
37
|
+
bool recursive;
|
38
|
+
int visits; // Current visits on the stack
|
39
|
+
VALUE source_file; // Source file
|
40
|
+
int source_line; // Line number
|
41
|
+
|
42
|
+
prof_measurement_t* measurement; // Stores measurement data for this method
|
43
|
+
} prof_method_t;
|
44
|
+
|
45
|
+
void rp_init_method_info(void);
|
46
|
+
|
47
|
+
st_data_t method_key(VALUE klass, VALUE msym);
|
48
|
+
|
49
|
+
st_table* method_table_create(void);
|
50
|
+
prof_method_t* method_table_lookup(st_table* table, st_data_t key);
|
51
|
+
size_t method_table_insert(st_table* table, st_data_t key, prof_method_t* val);
|
52
|
+
void method_table_free(st_table* table);
|
53
|
+
void prof_method_table_merge(st_table* self, st_table* other);
|
54
|
+
prof_method_t* prof_method_create(VALUE profile, VALUE klass, VALUE msym, VALUE source_file, int source_line);
|
55
|
+
prof_method_t* prof_get_method(VALUE self);
|
56
|
+
|
57
|
+
VALUE prof_method_wrap(prof_method_t* result);
|
58
|
+
void prof_method_mark(void* data);
|
59
|
+
|
60
|
+
VALUE resolve_klass(VALUE klass, unsigned int* klass_flags);
|
61
|
+
VALUE resolve_klass_name(VALUE klass, unsigned int* klass_flags);
|
62
|
+
|
63
|
+
#endif //__RP_METHOD_INFO__
|