2013-05-26 20:21:02 -04:00
|
|
|
#include "ruby/ruby.h"
|
|
|
|
#include "ruby/debug.h"
|
|
|
|
|
2013-09-22 16:10:40 -04:00
|
|
|
struct tracepoint_track {
|
|
|
|
size_t newobj_count;
|
|
|
|
size_t free_count;
|
|
|
|
size_t gc_start_count;
|
|
|
|
size_t gc_end_count;
|
|
|
|
size_t objects_count;
|
|
|
|
VALUE objects[10];
|
|
|
|
};
|
2013-05-26 20:21:02 -04:00
|
|
|
|
2013-09-22 16:10:40 -04:00
|
|
|
#define objects_max (sizeof(((struct tracepoint_track *)NULL)->objects)/sizeof(VALUE))
|
|
|
|
|
|
|
|
static void
|
2013-05-26 20:21:02 -04:00
|
|
|
tracepoint_track_objspace_events_i(VALUE tpval, void *data)
|
|
|
|
{
|
|
|
|
rb_trace_arg_t *tparg = rb_tracearg_from_tracepoint(tpval);
|
2013-09-22 16:10:40 -04:00
|
|
|
struct tracepoint_track *track = data;
|
|
|
|
|
2013-05-26 20:21:02 -04:00
|
|
|
switch (rb_tracearg_event_flag(tparg)) {
|
|
|
|
case RUBY_INTERNAL_EVENT_NEWOBJ:
|
|
|
|
{
|
|
|
|
VALUE obj = rb_tracearg_object(tparg);
|
2013-09-22 16:10:40 -04:00
|
|
|
if (track->objects_count < objects_max)
|
|
|
|
track->objects[track->objects_count++] = obj;
|
|
|
|
track->newobj_count++;
|
2013-05-26 20:21:02 -04:00
|
|
|
break;
|
|
|
|
}
|
2013-05-27 06:28:25 -04:00
|
|
|
case RUBY_INTERNAL_EVENT_FREEOBJ:
|
2013-05-26 20:21:02 -04:00
|
|
|
{
|
2013-09-22 16:10:40 -04:00
|
|
|
track->free_count++;
|
2013-05-26 20:21:02 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case RUBY_INTERNAL_EVENT_GC_START:
|
|
|
|
{
|
2013-09-22 16:10:40 -04:00
|
|
|
track->gc_start_count++;
|
2013-05-26 20:21:02 -04:00
|
|
|
break;
|
|
|
|
}
|
2013-05-27 11:40:27 -04:00
|
|
|
case RUBY_INTERNAL_EVENT_GC_END:
|
|
|
|
{
|
2013-09-22 16:10:40 -04:00
|
|
|
track->gc_end_count++;
|
2013-05-27 11:40:27 -04:00
|
|
|
break;
|
|
|
|
}
|
2013-05-26 20:21:02 -04:00
|
|
|
default:
|
|
|
|
rb_raise(rb_eRuntimeError, "unknown event");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-22 16:10:40 -04:00
|
|
|
static VALUE
|
2013-05-26 20:21:02 -04:00
|
|
|
tracepoint_track_objspace_events(VALUE self)
|
|
|
|
{
|
2013-09-22 16:10:40 -04:00
|
|
|
struct tracepoint_track track = {0, 0, 0, 0, 0, {}};
|
2013-05-27 11:40:27 -04:00
|
|
|
VALUE tpval = rb_tracepoint_new(0, RUBY_INTERNAL_EVENT_NEWOBJ | RUBY_INTERNAL_EVENT_FREEOBJ |
|
|
|
|
RUBY_INTERNAL_EVENT_GC_START | RUBY_INTERNAL_EVENT_GC_END,
|
2013-09-22 16:10:40 -04:00
|
|
|
tracepoint_track_objspace_events_i, &track);
|
2013-05-26 20:21:02 -04:00
|
|
|
VALUE result = rb_ary_new();
|
|
|
|
|
|
|
|
rb_tracepoint_enable(tpval);
|
|
|
|
rb_yield(Qundef);
|
|
|
|
rb_tracepoint_disable(tpval);
|
|
|
|
|
2013-09-22 16:10:40 -04:00
|
|
|
rb_ary_push(result, SIZET2NUM(track.newobj_count));
|
|
|
|
rb_ary_push(result, SIZET2NUM(track.free_count));
|
|
|
|
rb_ary_push(result, SIZET2NUM(track.gc_start_count));
|
|
|
|
rb_ary_push(result, SIZET2NUM(track.gc_end_count));
|
|
|
|
rb_ary_cat(result, track.objects, track.objects_count);
|
2013-05-26 20:21:02 -04:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Init_tracepoint(void)
|
|
|
|
{
|
|
|
|
VALUE mBug = rb_define_module("Bug");
|
|
|
|
rb_define_module_function(mBug, "tracepoint_track_objspace_events", tracepoint_track_objspace_events, 0);
|
|
|
|
}
|