summaryrefslogtreecommitdiff
path: root/vm_trace.c
diff options
context:
space:
mode:
authorko1 <ko1@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-12-06 13:42:32 +0000
committerko1 <ko1@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-12-06 13:42:32 +0000
commit5ac990e83eb4399b38542d788622c8069c7c4192 (patch)
tree3eaf556c26f92948bd2e1bd0c55ff3ef08b2010d /vm_trace.c
parent586e018c739d50be00bb7d750907d53009a0229b (diff)
`script_compiled` TracePoint event [Feature #15287]
* vm_trace.c: add `script_compiled` event. This event invoked after script compiling and before evaluating compiled script. Also the following methods are added: `TracePoint#compiled_instruction_sequence` method to get compiled `RubyVM::InstructionSequence` instance. `TracePoint#compiled_eval_script` method to get compiled script (String) by *eval methods (return nil if compiling by file). * vm_trace.c (tracepoint_attr_raised_exception): git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@66249 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'vm_trace.c')
-rw-r--r--vm_trace.c80
1 files changed, 80 insertions, 0 deletions
diff --git a/vm_trace.c b/vm_trace.c
index 23bba86d29..32df970ff9 100644
--- a/vm_trace.c
+++ b/vm_trace.c
@@ -629,6 +629,7 @@ get_event_id(rb_event_flag_t event)
C(thread_begin, THREAD_BEGIN);
C(thread_end, THREAD_END);
C(fiber_switch, FIBER_SWITCH);
+ C(script_compiled, SCRIPT_COMPILED);
#undef C
default:
return 0;
@@ -763,6 +764,9 @@ symbol2event_flag(VALUE v)
C(thread_begin, THREAD_BEGIN);
C(thread_end, THREAD_END);
C(fiber_switch, FIBER_SWITCH);
+ C(script_compiled, SCRIPT_COMPILED);
+
+ /* joke */
C(a_call, A_CALL);
C(a_return, A_RETURN);
#undef C
@@ -880,6 +884,7 @@ rb_tracearg_parameters(rb_trace_arg_t *trace_arg)
case RUBY_EVENT_LINE:
case RUBY_EVENT_CLASS:
case RUBY_EVENT_END:
+ case RUBY_EVENT_SCRIPT_COMPILED:
rb_raise(rb_eRuntimeError, "not supported by this event");
break;
}
@@ -958,6 +963,57 @@ rb_tracearg_raised_exception(rb_trace_arg_t *trace_arg)
}
VALUE
+rb_tracearg_compiled_eval_script(rb_trace_arg_t *trace_arg)
+{
+ VALUE data = trace_arg->data;
+
+ if (trace_arg->event & (RUBY_EVENT_SCRIPT_COMPILED)) {
+ /* ok */
+ }
+ else {
+ rb_raise(rb_eRuntimeError, "not supported by this event");
+ }
+ if (data == Qundef) {
+ rb_bug("rb_tracearg_raised_exception: unreachable");
+ }
+ if (rb_obj_is_iseq(data)) {
+ return Qnil;
+ }
+ else {
+ VM_ASSERT(RB_TYPE_P(data, T_ARRAY));
+ /* [src, iseq] */
+ return RARRAY_AREF(data, 0);
+ }
+}
+
+VALUE
+rb_tracearg_compiled_instruction_sequence(rb_trace_arg_t *trace_arg)
+{
+ VALUE data = trace_arg->data;
+
+ if (trace_arg->event & (RUBY_EVENT_SCRIPT_COMPILED)) {
+ /* ok */
+ }
+ else {
+ rb_raise(rb_eRuntimeError, "not supported by this event");
+ }
+ if (data == Qundef) {
+ rb_bug("rb_tracearg_raised_exception: unreachable");
+ }
+
+ if (rb_obj_is_iseq(data)) {
+ return rb_iseqw_new((const rb_iseq_t *)data);
+ }
+ else {
+ VM_ASSERT(RB_TYPE_P(data, T_ARRAY));
+ VM_ASSERT(rb_obj_is_iseq(RARRAY_AREF(data, 1)));
+
+ /* [src, iseq] */
+ return rb_iseqw_new((const rb_iseq_t *)RARRAY_AREF(data, 1));
+ }
+}
+
+VALUE
rb_tracearg_object(rb_trace_arg_t *trace_arg)
{
if (trace_arg->event & (RUBY_INTERNAL_EVENT_NEWOBJ | RUBY_INTERNAL_EVENT_FREEOBJ)) {
@@ -1107,6 +1163,28 @@ tracepoint_attr_raised_exception(VALUE tpval)
return rb_tracearg_raised_exception(get_trace_arg());
}
+/*
+ * Compiled source code (String) on *eval methods on the +:script_compiled+ event.
+ * If loaded from a file, it will return nil.
+ */
+static VALUE
+tracepoint_attr_compiled_eval_script(VALUE tpval)
+{
+ return rb_tracearg_compiled_eval_script(get_trace_arg());
+}
+
+/*
+ * Compiled instruction sequence represented by a RubyVM::InstructionSequence instance
+ * on the +:script_compiled+ event.
+ *
+ * Note that this method is MRI specific.
+ */
+static VALUE
+tracepoint_attr_compiled_instruction_sequence(VALUE tpval)
+{
+ return rb_tracearg_compiled_instruction_sequence(get_trace_arg());
+}
+
static void
tp_call_trace(VALUE tpval, rb_trace_arg_t *trace_arg)
{
@@ -1740,6 +1818,8 @@ Init_vm_trace(void)
rb_define_method(rb_cTracePoint, "self", tracepoint_attr_self, 0);
rb_define_method(rb_cTracePoint, "return_value", tracepoint_attr_return_value, 0);
rb_define_method(rb_cTracePoint, "raised_exception", tracepoint_attr_raised_exception, 0);
+ rb_define_method(rb_cTracePoint, "compiled_eval_script", tracepoint_attr_compiled_eval_script, 0);
+ rb_define_method(rb_cTracePoint, "compiled_instruction_sequence", tracepoint_attr_compiled_instruction_sequence, 0);
rb_define_singleton_method(rb_cTracePoint, "stat", tracepoint_stat_s, 0);
}