ruby/ext/-test-/debug/profile_frames.c
Daisuke Aritomo 4adf418be9 [Feature #10602] Add new API rb_profile_thread_frames()
Add a new API rb_profile_thread_frames(), which is essentialy a
per-thread version of rb_profile_frames().

While the original rb_profile_frames() always returns results about the
current active thread obtained by GET_EC(), this new API takes a Thread
to be profiled as an argument.

This should come in handy when profiling I/O-bound programs such as
webapps, since this new API allows us to learn about Threads performing
I/O (which do not have the GVL).

Profiling worker threads (such as Sidekiq workers) may be another
application.

Implements [Feature #10602]

Co-authored-by: Mike Perham <mike@perham.net>
2023-10-31 11:16:18 +09:00

65 lines
2.1 KiB
C

#include "ruby/ruby.h"
#include "ruby/debug.h"
#define MAX_BUF_SIZE 0x100
static VALUE
profile_frames(VALUE self, VALUE start_v, VALUE num_v)
{
int i, collected_size;
int start = NUM2INT(start_v);
int buff_size = NUM2INT(num_v);
VALUE buff[MAX_BUF_SIZE];
int lines[MAX_BUF_SIZE];
VALUE result = rb_ary_new();
if (buff_size > MAX_BUF_SIZE) rb_raise(rb_eRuntimeError, "too long buff_size");
collected_size = rb_profile_frames(start, buff_size, buff, lines);
for (i=0; i<collected_size; i++) {
VALUE ary = rb_ary_new();
rb_ary_push(ary, rb_profile_frame_path(buff[i]));
rb_ary_push(ary, rb_profile_frame_absolute_path(buff[i]));
rb_ary_push(ary, rb_profile_frame_label(buff[i]));
rb_ary_push(ary, rb_profile_frame_base_label(buff[i]));
rb_ary_push(ary, rb_profile_frame_full_label(buff[i]));
rb_ary_push(ary, rb_profile_frame_first_lineno(buff[i]));
rb_ary_push(ary, rb_profile_frame_classpath(buff[i]));
rb_ary_push(ary, rb_profile_frame_singleton_method_p(buff[i]));
rb_ary_push(ary, rb_profile_frame_method_name(buff[i]));
rb_ary_push(ary, rb_profile_frame_qualified_method_name(buff[i]));
rb_ary_push(ary, INT2NUM(lines[i]));
rb_ary_push(result, ary);
}
return result;
}
static VALUE
profile_thread_frames(VALUE self, VALUE thread, VALUE start_v, VALUE num_v)
{
int i, collected_size;
int start = NUM2INT(start_v);
int buff_size = NUM2INT(num_v);
VALUE buff[MAX_BUF_SIZE];
int lines[MAX_BUF_SIZE];
VALUE result = rb_ary_new();
if (buff_size > MAX_BUF_SIZE) rb_raise(rb_eRuntimeError, "too long buff_size");
collected_size = rb_profile_thread_frames(thread, start, buff_size, buff, lines);
for (i=0; i<collected_size; i++) {
rb_ary_push(result, rb_profile_frame_full_label(buff[i]));
}
return result;
}
void
Init_profile_frames(VALUE klass)
{
rb_define_module_function(klass, "profile_frames", profile_frames, 2);
rb_define_module_function(klass, "profile_thread_frames", profile_thread_frames, 3);
}