Make tracepoints with set_trace_func or TracePoint.new ractor local (#15468)

Before this change, GC'ing any Ractor object caused you to lose all
enabled tracepoints across all ractors (even main). Now tracepoints are
ractor-local and this doesn't happen. Internal events are still global.

Fixes [Bug #19112]
This commit is contained in:
Luke Gruber 2025-12-16 14:06:55 -05:00 committed by GitHub
parent d209e6f1c0
commit 4fb537b1ee
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
Notes: git 2025-12-16 19:07:24 +00:00
Merged-By: luke-gru <luke.gru@gmail.com>
16 changed files with 694 additions and 226 deletions

2
depend
View File

@ -7606,6 +7606,7 @@ iseq.$(OBJEXT): {$(VPATH)}onigmo.h
iseq.$(OBJEXT): {$(VPATH)}oniguruma.h
iseq.$(OBJEXT): {$(VPATH)}prism_compile.h
iseq.$(OBJEXT): {$(VPATH)}ractor.h
iseq.$(OBJEXT): {$(VPATH)}ractor_core.h
iseq.$(OBJEXT): {$(VPATH)}ruby_assert.h
iseq.$(OBJEXT): {$(VPATH)}ruby_atomic.h
iseq.$(OBJEXT): {$(VPATH)}rubyparser.h
@ -19760,6 +19761,7 @@ vm_trace.$(OBJEXT): {$(VPATH)}onigmo.h
vm_trace.$(OBJEXT): {$(VPATH)}oniguruma.h
vm_trace.$(OBJEXT): {$(VPATH)}prism_compile.h
vm_trace.$(OBJEXT): {$(VPATH)}ractor.h
vm_trace.$(OBJEXT): {$(VPATH)}ractor_core.h
vm_trace.$(OBJEXT): {$(VPATH)}ruby_assert.h
vm_trace.$(OBJEXT): {$(VPATH)}ruby_atomic.h
vm_trace.$(OBJEXT): {$(VPATH)}rubyparser.h

View File

@ -306,9 +306,6 @@ mark_and_move_method_entry(rb_method_entry_t *ment, bool reference_updating)
if (!rb_gc_checking_shareable()) {
rb_gc_mark_and_move(&def->body.bmethod.proc);
}
if (def->body.bmethod.hooks) {
rb_hook_list_mark_and_move(def->body.bmethod.hooks);
}
break;
case VM_METHOD_TYPE_ALIAS:
rb_gc_mark_and_move_ptr(&def->body.alias.original_me);

141
iseq.c
View File

@ -39,6 +39,7 @@
#include "iseq.h"
#include "ruby/util.h"
#include "vm_core.h"
#include "ractor_core.h"
#include "vm_callinfo.h"
#include "yjit.h"
#include "ruby/ractor.h"
@ -161,6 +162,24 @@ iseq_clear_ic_references(const rb_iseq_t *iseq)
}
}
rb_hook_list_t *
rb_iseq_local_hooks(const rb_iseq_t *iseq, rb_ractor_t *r, bool create)
{
rb_hook_list_t *hook_list = NULL;
st_data_t val;
if (st_lookup(rb_ractor_targeted_hooks(r), (st_data_t)iseq, &val)) {
hook_list = (rb_hook_list_t*)val;
RUBY_ASSERT(hook_list->type == hook_list_type_targeted_iseq);
}
else if (create) {
hook_list = RB_ZALLOC(rb_hook_list_t);
hook_list->type = hook_list_type_targeted_iseq;
st_insert(rb_ractor_targeted_hooks(r), (st_data_t)iseq, (st_data_t)hook_list);
}
return hook_list;
}
void
rb_iseq_free(const rb_iseq_t *iseq)
{
@ -213,10 +232,6 @@ rb_iseq_free(const rb_iseq_t *iseq)
ruby_xfree(body);
}
if (iseq && ISEQ_EXECUTABLE_P(iseq) && iseq->aux.exec.local_hooks) {
rb_hook_list_free(iseq->aux.exec.local_hooks);
}
RUBY_FREE_LEAVE("iseq");
}
@ -448,10 +463,6 @@ rb_iseq_mark_and_move(rb_iseq_t *iseq, bool reference_updating)
else {
/* executable */
VM_ASSERT(ISEQ_EXECUTABLE_P(iseq));
if (iseq->aux.exec.local_hooks) {
rb_hook_list_mark_and_move(iseq->aux.exec.local_hooks);
}
}
RUBY_MARK_LEAVE("iseq");
@ -2438,17 +2449,22 @@ rb_iseq_event_flags(const rb_iseq_t *iseq, size_t pos)
}
}
static void rb_iseq_trace_flag_cleared(const rb_iseq_t *iseq, size_t pos);
// Clear tracing event flags and turn off tracing for a given instruction as needed.
// This is currently used after updating a one-shot line coverage for the current instruction.
void
rb_iseq_clear_event_flags(const rb_iseq_t *iseq, size_t pos, rb_event_flag_t reset)
{
struct iseq_insn_info_entry *entry = (struct iseq_insn_info_entry *)get_insn_info(iseq, pos);
if (entry) {
entry->events &= ~reset;
if (!(entry->events & iseq->aux.exec.global_trace_events)) {
void rb_iseq_trace_flag_cleared(const rb_iseq_t *iseq, size_t pos);
rb_iseq_trace_flag_cleared(iseq, pos);
RB_VM_LOCKING() {
rb_vm_barrier();
struct iseq_insn_info_entry *entry = (struct iseq_insn_info_entry *)get_insn_info(iseq, pos);
if (entry) {
entry->events &= ~reset;
if (!(entry->events & iseq->aux.exec.global_trace_events)) {
rb_iseq_trace_flag_cleared(iseq, pos);
}
}
}
}
@ -3930,14 +3946,15 @@ rb_vm_insn_decode(const VALUE encoded)
// Turn on or off tracing for a given instruction address
static inline int
encoded_iseq_trace_instrument(VALUE *iseq_encoded_insn, rb_event_flag_t turnon, bool remain_current_trace)
encoded_iseq_trace_instrument(VALUE *iseq_encoded_insn, rb_event_flag_t turnon, bool remain_traced)
{
ASSERT_vm_locking();
st_data_t key = (st_data_t)*iseq_encoded_insn;
st_data_t val;
if (st_lookup(encoded_insn_data, key, &val)) {
insn_data_t *e = (insn_data_t *)val;
if (remain_current_trace && key == (st_data_t)e->trace_encoded_insn) {
if (remain_traced && key == (st_data_t)e->trace_encoded_insn) {
turnon = 1;
}
*iseq_encoded_insn = (VALUE) (turnon ? e->trace_encoded_insn : e->notrace_encoded_insn);
@ -3948,7 +3965,7 @@ encoded_iseq_trace_instrument(VALUE *iseq_encoded_insn, rb_event_flag_t turnon,
}
// Turn off tracing for an instruction at pos after tracing event flags are cleared
void
static void
rb_iseq_trace_flag_cleared(const rb_iseq_t *iseq, size_t pos)
{
const struct rb_iseq_constant_body *const body = ISEQ_BODY(iseq);
@ -3974,14 +3991,16 @@ add_bmethod_events(rb_event_flag_t events)
// Note, to support call/return events for bmethods, turnon_event can have more events than tpval.
static int
iseq_add_local_tracepoint(const rb_iseq_t *iseq, rb_event_flag_t turnon_events, VALUE tpval, unsigned int target_line)
iseq_add_local_tracepoint(const rb_iseq_t *iseq, rb_event_flag_t turnon_events, VALUE tpval, unsigned int target_line, rb_ractor_t *r)
{
unsigned int pc;
int n = 0;
const struct rb_iseq_constant_body *const body = ISEQ_BODY(iseq);
VALUE *iseq_encoded = (VALUE *)body->iseq_encoded;
rb_iseq_t *iseq_mut = (rb_iseq_t*)iseq;
VM_ASSERT(ISEQ_EXECUTABLE_P(iseq));
ASSERT_vm_locking_with_barrier();
for (pc=0; pc<body->iseq_size;) {
const struct iseq_insn_info_entry *entry = get_insn_info(iseq, pc);
@ -4003,11 +4022,9 @@ iseq_add_local_tracepoint(const rb_iseq_t *iseq, rb_event_flag_t turnon_events,
}
if (n > 0) {
if (iseq->aux.exec.local_hooks == NULL) {
((rb_iseq_t *)iseq)->aux.exec.local_hooks = RB_ZALLOC(rb_hook_list_t);
iseq->aux.exec.local_hooks->is_local = true;
}
rb_hook_list_connect_tracepoint((VALUE)iseq, iseq->aux.exec.local_hooks, tpval, target_line);
rb_hook_list_t *hook_list = rb_iseq_local_hooks(iseq, r, true);
rb_hook_list_connect_local_tracepoint(hook_list, tpval, target_line);
iseq_mut->aux.exec.local_hooks_cnt++;
}
return n;
@ -4018,19 +4035,21 @@ struct trace_set_local_events_struct {
VALUE tpval;
unsigned int target_line;
int n;
rb_ractor_t *r;
};
static void
iseq_add_local_tracepoint_i(const rb_iseq_t *iseq, void *p)
{
struct trace_set_local_events_struct *data = (struct trace_set_local_events_struct *)p;
data->n += iseq_add_local_tracepoint(iseq, data->turnon_events, data->tpval, data->target_line);
data->n += iseq_add_local_tracepoint(iseq, data->turnon_events, data->tpval, data->target_line, data->r);
iseq_iterate_children(iseq, iseq_add_local_tracepoint_i, p);
}
int
rb_iseq_add_local_tracepoint_recursively(const rb_iseq_t *iseq, rb_event_flag_t turnon_events, VALUE tpval, unsigned int target_line, bool target_bmethod)
{
ASSERT_vm_locking_with_barrier();
struct trace_set_local_events_struct data;
if (target_bmethod) {
turnon_events = add_bmethod_events(turnon_events);
@ -4039,35 +4058,52 @@ rb_iseq_add_local_tracepoint_recursively(const rb_iseq_t *iseq, rb_event_flag_t
data.tpval = tpval;
data.target_line = target_line;
data.n = 0;
data.r = GET_RACTOR();
iseq_add_local_tracepoint_i(iseq, (void *)&data);
if (0) rb_funcall(Qnil, rb_intern("puts"), 1, rb_iseq_disasm(iseq)); /* for debug */
if (0) fprintf(stderr, "Iseq disasm:\n:%s", RSTRING_PTR(rb_iseq_disasm(iseq))); /* for debug */
return data.n;
}
static int
iseq_remove_local_tracepoint(const rb_iseq_t *iseq, VALUE tpval)
iseq_remove_local_tracepoint(const rb_iseq_t *iseq, VALUE tpval, rb_ractor_t *r)
{
int n = 0;
unsigned int num_hooks_left;
unsigned int pc;
const struct rb_iseq_constant_body *body;
rb_iseq_t *iseq_mut = (rb_iseq_t*)iseq;
rb_hook_list_t *hook_list;
VALUE *iseq_encoded;
ASSERT_vm_locking_with_barrier();
if (iseq->aux.exec.local_hooks) {
unsigned int pc;
const struct rb_iseq_constant_body *const body = ISEQ_BODY(iseq);
VALUE *iseq_encoded = (VALUE *)body->iseq_encoded;
hook_list = rb_iseq_local_hooks(iseq, r, false);
if (hook_list) {
rb_event_flag_t local_events = 0;
rb_hook_list_remove_tracepoint(iseq->aux.exec.local_hooks, tpval);
local_events = iseq->aux.exec.local_hooks->events;
rb_event_flag_t prev_events = hook_list->events;
if (rb_hook_list_remove_local_tracepoint(hook_list, tpval)) {
RUBY_ASSERT(iseq->aux.exec.local_hooks_cnt > 0);
iseq_mut->aux.exec.local_hooks_cnt--;
local_events = hook_list->events; // remaining events for this ractor
num_hooks_left = rb_hook_list_count(hook_list);
if (local_events == 0 && prev_events != 0) {
st_delete(rb_ractor_targeted_hooks(r), (st_data_t*)&iseq, NULL);
rb_hook_list_free(hook_list);
}
if (local_events == 0) {
rb_hook_list_free(iseq->aux.exec.local_hooks);
((rb_iseq_t *)iseq)->aux.exec.local_hooks = NULL;
}
if (iseq->aux.exec.local_hooks_cnt == num_hooks_left) {
body = ISEQ_BODY(iseq);
iseq_encoded = (VALUE *)body->iseq_encoded;
local_events = add_bmethod_events(local_events);
for (pc = 0; pc<body->iseq_size;) {
rb_event_flag_t pc_events = rb_iseq_event_flags(iseq, pc);
pc += encoded_iseq_trace_instrument(&iseq_encoded[pc], pc_events & (local_events | iseq->aux.exec.global_trace_events), false);
}
}
local_events = add_bmethod_events(local_events);
for (pc = 0; pc<body->iseq_size;) {
rb_event_flag_t pc_events = rb_iseq_event_flags(iseq, pc);
pc += encoded_iseq_trace_instrument(&iseq_encoded[pc], pc_events & (local_events | iseq->aux.exec.global_trace_events), false);
n++;
}
}
return n;
@ -4076,22 +4112,25 @@ iseq_remove_local_tracepoint(const rb_iseq_t *iseq, VALUE tpval)
struct trace_clear_local_events_struct {
VALUE tpval;
int n;
rb_ractor_t *r;
};
static void
iseq_remove_local_tracepoint_i(const rb_iseq_t *iseq, void *p)
{
struct trace_clear_local_events_struct *data = (struct trace_clear_local_events_struct *)p;
data->n += iseq_remove_local_tracepoint(iseq, data->tpval);
data->n += iseq_remove_local_tracepoint(iseq, data->tpval, data->r);
iseq_iterate_children(iseq, iseq_remove_local_tracepoint_i, p);
}
int
rb_iseq_remove_local_tracepoint_recursively(const rb_iseq_t *iseq, VALUE tpval)
rb_iseq_remove_local_tracepoint_recursively(const rb_iseq_t *iseq, VALUE tpval, rb_ractor_t *r)
{
struct trace_clear_local_events_struct data;
ASSERT_vm_locking_with_barrier();
data.tpval = tpval;
data.n = 0;
data.r = r;
iseq_remove_local_tracepoint_i(iseq, (void *)&data);
return data.n;
@ -4109,11 +4148,14 @@ rb_iseq_trace_set(const rb_iseq_t *iseq, rb_event_flag_t turnon_events)
return;
}
else {
// NOTE: this does not need VM barrier if it's a new ISEQ
unsigned int pc;
const struct rb_iseq_constant_body *const body = ISEQ_BODY(iseq);
VALUE *iseq_encoded = (VALUE *)body->iseq_encoded;
rb_event_flag_t enabled_events;
rb_event_flag_t local_events = iseq->aux.exec.local_hooks ? iseq->aux.exec.local_hooks->events : 0;
rb_hook_list_t *local_hooks = rb_iseq_local_hooks(iseq, GET_RACTOR(), false);
rb_event_flag_t local_events = local_hooks ? local_hooks->events : 0;
((rb_iseq_t *)iseq)->aux.exec.global_trace_events = turnon_events;
enabled_events = add_bmethod_events(turnon_events | local_events);
@ -4129,6 +4171,7 @@ void rb_vm_cc_general(const struct rb_callcache *cc);
static bool
clear_attr_cc(VALUE v)
{
ASSERT_vm_locking_with_barrier();
if (imemo_type_p(v, imemo_callcache) && vm_cc_ivar_p((const struct rb_callcache *)v)) {
rb_vm_cc_general((struct rb_callcache *)v);
return true;
@ -4141,6 +4184,7 @@ clear_attr_cc(VALUE v)
static bool
clear_bf_cc(VALUE v)
{
ASSERT_vm_locking_with_barrier();
if (imemo_type_p(v, imemo_callcache) && vm_cc_bf_p((const struct rb_callcache *)v)) {
rb_vm_cc_general((struct rb_callcache *)v);
return true;
@ -4166,7 +4210,10 @@ clear_attr_ccs_i(void *vstart, void *vend, size_t stride, void *data)
void
rb_clear_attr_ccs(void)
{
rb_objspace_each_objects(clear_attr_ccs_i, NULL);
RB_VM_LOCKING() {
rb_vm_barrier();
rb_objspace_each_objects(clear_attr_ccs_i, NULL);
}
}
static int
@ -4185,6 +4232,7 @@ clear_bf_ccs_i(void *vstart, void *vend, size_t stride, void *data)
void
rb_clear_bf_ccs(void)
{
ASSERT_vm_locking_with_barrier();
rb_objspace_each_objects(clear_bf_ccs_i, NULL);
}
@ -4214,7 +4262,10 @@ trace_set_i(void *vstart, void *vend, size_t stride, void *data)
void
rb_iseq_trace_set_all(rb_event_flag_t turnon_events)
{
rb_objspace_each_objects(trace_set_i, &turnon_events);
RB_VM_LOCKING() {
rb_vm_barrier();
rb_objspace_each_objects(trace_set_i, &turnon_events);
}
}
VALUE

4
iseq.h
View File

@ -190,10 +190,12 @@ const rb_iseq_t *rb_iseq_ibf_load_bytes(const char *cstr, size_t);
VALUE rb_iseq_ibf_load_extra_data(VALUE str);
void rb_iseq_init_trace(rb_iseq_t *iseq);
int rb_iseq_add_local_tracepoint_recursively(const rb_iseq_t *iseq, rb_event_flag_t turnon_events, VALUE tpval, unsigned int target_line, bool target_bmethod);
int rb_iseq_remove_local_tracepoint_recursively(const rb_iseq_t *iseq, VALUE tpval);
int rb_iseq_remove_local_tracepoint_recursively(const rb_iseq_t *iseq, VALUE tpval, rb_ractor_t *r);
const rb_iseq_t *rb_iseq_load_iseq(VALUE fname);
const rb_iseq_t *rb_iseq_compile_iseq(VALUE str, VALUE fname);
int rb_iseq_opt_frozen_string_literal(void);
rb_hook_list_t *rb_iseq_local_hooks(const rb_iseq_t *iseq, rb_ractor_t *r, bool create);
#if VM_INSN_INFO_TABLE_IMPL == 2
unsigned int *rb_iseq_insns_info_decode_positions(const struct rb_iseq_constant_body *body);

View File

@ -166,8 +166,8 @@ typedef struct rb_method_refined_struct {
typedef struct rb_method_bmethod_struct {
VALUE proc; /* should be marked */
struct rb_hook_list_struct *hooks;
rb_serial_t defined_ractor_id;
unsigned int local_hooks_cnt;
} rb_method_bmethod_t;
enum method_optimized_type {
@ -208,6 +208,8 @@ struct rb_method_definition_struct {
};
struct rb_id_table;
struct rb_ractor_struct;
struct rb_hook_list_struct;
typedef struct rb_method_definition_struct rb_method_definition_t;
STATIC_ASSERT(sizeof_method_def, offsetof(rb_method_definition_t, body) <= 8);
@ -267,5 +269,8 @@ void rb_vm_delete_cc_refinement(const struct rb_callcache *cc);
void rb_clear_method_cache(VALUE klass_or_module, ID mid);
void rb_clear_all_refinement_method_cache(void);
void rb_invalidate_method_caches(struct rb_id_table *cm_tbl, VALUE cc_tbl);
struct rb_hook_list_struct *rb_method_def_local_hooks(rb_method_definition_t *def, struct rb_ractor_struct *cr, bool create);
void rb_method_definition_addref(rb_method_definition_t *def);
void rb_method_definition_release(rb_method_definition_t *def);
#endif /* RUBY_METHOD_H */

View File

@ -207,6 +207,24 @@ static void ractor_sync_free(rb_ractor_t *r);
static size_t ractor_sync_memsize(const rb_ractor_t *r);
static void ractor_sync_init(rb_ractor_t *r);
static int
mark_targeted_hook_list(st_data_t key, st_data_t value, st_data_t _arg)
{
rb_hook_list_t *hook_list = (rb_hook_list_t*)value;
if (hook_list->type == hook_list_type_targeted_iseq) {
rb_gc_mark((VALUE)key);
}
else {
rb_method_definition_t *def = (rb_method_definition_t*)key;
RUBY_ASSERT(hook_list->type == hook_list_type_targeted_def);
rb_gc_mark(def->body.bmethod.proc);
}
rb_hook_list_mark(hook_list);
return ST_CONTINUE;
}
static void
ractor_mark(void *ptr)
{
@ -228,6 +246,9 @@ ractor_mark(void *ptr)
ractor_sync_mark(r);
rb_hook_list_mark(&r->pub.hooks);
if (r->pub.targeted_hooks) {
st_foreach(r->pub.targeted_hooks, mark_targeted_hook_list, 0);
}
if (r->threads.cnt > 0) {
rb_thread_t *th = 0;
@ -241,17 +262,33 @@ ractor_mark(void *ptr)
}
}
static int
free_targeted_hook_lists(st_data_t key, st_data_t val, st_data_t _arg)
{
rb_hook_list_t *hook_list = (rb_hook_list_t*)val;
rb_hook_list_free(hook_list);
return ST_DELETE;
}
static void
free_targeted_hooks(st_table *hooks_tbl)
{
st_foreach(hooks_tbl, free_targeted_hook_lists, 0);
}
static void
ractor_free(void *ptr)
{
rb_ractor_t *r = (rb_ractor_t *)ptr;
RUBY_DEBUG_LOG("free r:%d", rb_ractor_id(r));
free_targeted_hooks(r->pub.targeted_hooks);
rb_native_mutex_destroy(&r->sync.lock);
#ifdef RUBY_THREAD_WIN32_H
rb_native_cond_destroy(&r->sync.wakeup_cond);
#endif
ractor_local_storage_free(r);
rb_hook_list_free(&r->pub.hooks);
st_free_table(r->pub.targeted_hooks);
if (r->newobj_cache) {
RUBY_ASSERT(r == ruby_single_main_ractor);
@ -489,6 +526,8 @@ static void
ractor_init(rb_ractor_t *r, VALUE name, VALUE loc)
{
ractor_sync_init(r);
r->pub.targeted_hooks = st_init_numtable();
r->pub.hooks.type = hook_list_type_ractor_local;
// thread management
rb_thread_sched_init(&r->threads.sched, false);
@ -1136,6 +1175,12 @@ rb_ractor_hooks(rb_ractor_t *cr)
return &cr->pub.hooks;
}
st_table *
rb_ractor_targeted_hooks(rb_ractor_t *cr)
{
return cr->pub.targeted_hooks;
}
static void
rb_obj_set_shareable_no_assert(VALUE obj)
{

View File

@ -144,6 +144,7 @@ VALUE rb_ractor_require(VALUE feature, bool silent);
VALUE rb_ractor_autoload_load(VALUE space, ID id);
VALUE rb_ractor_ensure_shareable(VALUE obj, VALUE name);
st_table *rb_ractor_targeted_hooks(rb_ractor_t *cr);
RUBY_SYMBOL_EXPORT_BEGIN
void rb_ractor_finish_marking(void);
@ -250,6 +251,25 @@ rb_ractor_id(const rb_ractor_t *r)
return r->pub.id;
}
static inline void
rb_ractor_targeted_hooks_incr(rb_ractor_t *cr)
{
cr->pub.targeted_hooks_cnt++;
}
static inline void
rb_ractor_targeted_hooks_decr(rb_ractor_t *cr)
{
RUBY_ASSERT(cr->pub.targeted_hooks_cnt > 0);
cr->pub.targeted_hooks_cnt--;
}
static inline unsigned int
rb_ractor_targeted_hooks_cnt(rb_ractor_t *cr)
{
return cr->pub.targeted_hooks_cnt;
}
#if RACTOR_CHECK_MODE > 0
# define RACTOR_BELONGING_ID(obj) (*(uint32_t *)(((uintptr_t)(obj)) + rb_gc_obj_slot_size(obj)))

View File

@ -2957,4 +2957,210 @@ CODE
assert_kind_of(Thread, target_thread)
end
def test_tracepoint_garbage_collected_when_disable
before_count_stat = 0
before_count_objspace = 0
TracePoint.stat.each do
before_count_stat += 1
end
ObjectSpace.each_object(TracePoint) do
before_count_objspace += 1
end
tp = TracePoint.new(:c_call, :c_return) do
end
tp.enable
Class.inspect # c_call, c_return invoked
tp.disable
tp_id = tp.object_id
tp = nil
gc_times = 0
gc_max_retries = 10
EnvUtil.suppress_warning do
until (ObjectSpace._id2ref(tp_id) rescue nil).nil?
GC.start
gc_times += 1
if gc_times == gc_max_retries
break
end
end
end
return if gc_times == gc_max_retries
after_count_stat = 0
TracePoint.stat.each do |v|
after_count_stat += 1
end
assert after_count_stat <= before_count_stat
after_count_objspace = 0
ObjectSpace.each_object(TracePoint) do
after_count_objspace += 1
end
assert after_count_objspace <= before_count_objspace
end
def test_tp_ractor_local_untargeted
assert_ractor("#{<<~"begin;"}\n#{<<~'end;'}")
begin;
r = Ractor.new do
results = []
tp = TracePoint.new(:line) { |tp| results << tp.path }
tp.enable
Ractor.main << :continue
Ractor.receive
tp.disable
results
end
outer_results = []
outer_tp = TracePoint.new(:line) { |tp| outer_results << tp.path }
outer_tp.enable
Ractor.receive
GC.start # so I can check <internal:gc> path
r << :continue
inner_results = r.value
outer_tp.disable
assert_equal 1, outer_results.select { |path| path.match?(/internal:gc/) }.size
assert_equal 0, inner_results.select { |path| path.match?(/internal:gc/) }.size
end;
end
def test_tp_targeted_ractor_local_bmethod
assert_ractor("#{<<~"begin;"}\n#{<<~'end;'}")
begin;
mname = :foo
prok = Ractor.shareable_proc do
end
klass = EnvUtil.labeled_class(:Klass) do
define_method(mname, &prok)
end
outer_results = 0
_outer_tp = TracePoint.new(:call) do
outer_results += 1
end # not enabled
rs = 10.times.map do
Ractor.new(mname, klass) do |mname, klass0|
inner_results = 0
tp = TracePoint.new(:call) { |tp| inner_results += 1 }
target = klass0.instance_method(mname)
tp.enable(target: target)
obj = klass0.new
10.times { obj.send(mname) }
tp.disable
inner_results
end
end
inner_results = rs.map(&:value).sum
obj = klass.new
10.times { obj.send(mname) }
assert_equal 100, inner_results
assert_equal 0, outer_results
end;
end
def test_tp_targeted_ractor_local_method
assert_ractor("#{<<~"begin;"}\n#{<<~'end;'}")
begin;
def foo
end
outer_results = 0
_outer_tp = TracePoint.new(:call) do
outer_results += 1
end # not enabled
rs = 10.times.map do
Ractor.new do
inner_results = 0
tp = TracePoint.new(:call) do
inner_results += 1
end
tp.enable(target: method(:foo))
10.times { foo }
tp.disable
inner_results
end
end
inner_results = rs.map(&:value).sum
10.times { foo }
assert_equal 100, inner_results
assert_equal 0, outer_results
end;
end
def test_tracepoints_not_disabled_by_ractor_gc
assert_ractor("#{<<~"begin;"}\n#{<<~'end;'}")
begin;
$-w = nil # uses ObjectSpace._id2ref
def hi = "hi"
greetings = 0
tp_target = TracePoint.new(:call) do |tp|
greetings += 1
end
tp_target.enable(target: method(:hi))
raises = 0
tp_global = TracePoint.new(:raise) do |tp|
raises += 1
end
tp_global.enable
r = Ractor.new { 10 }
r.join
ractor_id = r.object_id
r = nil # allow gc for ractor
gc_max_retries = 15
gc_times = 0
# force GC of ractor (or try, because we have a conservative GC)
until (ObjectSpace._id2ref(ractor_id) rescue nil).nil?
GC.start
gc_times += 1
if gc_times == gc_max_retries
break
end
end
# tracepoints should still be enabled after GC of `r`
5.times {
hi
}
6.times {
raise "uh oh" rescue nil
}
tp_target.disable
tp_global.disable
assert_equal 5, greetings
if gc_times == gc_max_retries # _id2ref never raised
assert_equal 6, raises
else
assert_equal 7, raises
end
end;
end
def test_lots_of_enabled_tracepoints_ractor_gc
assert_ractor("#{<<~"begin;"}\n#{<<~'end;'}")
begin;
def foo; end
sum = 8.times.map do
Ractor.new do
called = 0
TracePoint.new(:call) do |tp|
next if tp.callee_id != :foo
called += 1
end.enable
200.times do
TracePoint.new(:line) {
# all these allocations shouldn't GC these tracepoints while the ractor is alive.
Object.new
}.enable
end
100.times { foo }
called
end
end.map(&:value).sum
assert_equal 800, sum
4.times { GC.start } # Now the tracepoints can be GC'd because the ractors can be GC'd
end;
end
end

View File

@ -26,7 +26,7 @@
# change. Instead, it is recommended to specify the types of events you
# want to use.
#
# To filter what is traced, you can pass any of the following as +events+:
# To filter what is traced, you can pass any number of the following as +events+:
#
# +:line+:: Execute an expression or statement on a new line.
# +:class+:: Start a class or module definition.
@ -74,7 +74,7 @@ class TracePoint
#
# A block must be given; otherwise, an ArgumentError is raised.
#
# If the trace method isn't included in the given events filter, a
# If the trace method isn't supported for the given event(s) filter, a
# RuntimeError is raised.
#
# TracePoint.trace(:line) do |tp|
@ -89,7 +89,9 @@ class TracePoint
# end
# $tp.lineno #=> access from outside (RuntimeError)
#
# Access from other threads is also forbidden.
# Access from other ractors, threads or fibers is forbidden. TracePoints are active
# per-ractor so if you enable a TracePoint in one ractor, other ractors will not be
# affected.
#
def self.new(*events)
Primitive.attr! :use_block

33
vm.c
View File

@ -718,9 +718,10 @@ rb_current_ec_noinline(void)
#endif
rb_event_flag_t ruby_vm_event_flags;
rb_event_flag_t ruby_vm_event_enabled_global_flags;
unsigned int ruby_vm_event_local_num;
rb_event_flag_t ruby_vm_event_flags = 0;
rb_event_flag_t ruby_vm_event_enabled_global_flags = 0;
unsigned int ruby_vm_c_events_enabled = 0;
unsigned int ruby_vm_iseq_events_enabled = 0;
rb_serial_t ruby_vm_constant_cache_invalidations = 0;
rb_serial_t ruby_vm_constant_cache_misses = 0;
@ -2579,7 +2580,11 @@ hook_before_rewind(rb_execution_context_t *ec, bool cfp_returning_with_value, in
}
else {
const rb_iseq_t *iseq = ec->cfp->iseq;
rb_hook_list_t *local_hooks = iseq->aux.exec.local_hooks;
rb_hook_list_t *local_hooks = NULL;
unsigned int local_hooks_cnt = iseq->aux.exec.local_hooks_cnt;
if (RB_UNLIKELY(local_hooks_cnt > 0)) {
local_hooks = rb_iseq_local_hooks(iseq, rb_ec_ractor_ptr(ec), false);
}
switch (VM_FRAME_TYPE(ec->cfp)) {
case VM_FRAME_MAGIC_METHOD:
@ -2617,15 +2622,18 @@ hook_before_rewind(rb_execution_context_t *ec, bool cfp_returning_with_value, in
bmethod_return_value);
VM_ASSERT(me->def->type == VM_METHOD_TYPE_BMETHOD);
local_hooks = me->def->body.bmethod.hooks;
if (UNLIKELY(local_hooks && local_hooks->events & RUBY_EVENT_RETURN)) {
rb_exec_event_hook_orig(ec, local_hooks, RUBY_EVENT_RETURN, ec->cfp->self,
rb_vm_frame_method_entry(ec->cfp)->def->original_id,
rb_vm_frame_method_entry(ec->cfp)->called_id,
rb_vm_frame_method_entry(ec->cfp)->owner,
bmethod_return_value, TRUE);
unsigned int local_hooks_cnt = me->def->body.bmethod.local_hooks_cnt;
if (UNLIKELY(local_hooks_cnt > 0)) {
local_hooks = rb_method_def_local_hooks(me->def, rb_ec_ractor_ptr(ec), false);
if (local_hooks && local_hooks->events & RUBY_EVENT_RETURN) {
rb_exec_event_hook_orig(ec, local_hooks, RUBY_EVENT_RETURN, ec->cfp->self,
rb_vm_frame_method_entry(ec->cfp)->def->original_id,
rb_vm_frame_method_entry(ec->cfp)->called_id,
rb_vm_frame_method_entry(ec->cfp)->owner,
bmethod_return_value, TRUE);
}
}
THROW_DATA_CONSUMED_SET(err);
}
else {
@ -4580,6 +4588,7 @@ Init_BareVM(void)
vm->overloaded_cme_table = st_init_numtable();
vm->constant_cache = rb_id_table_create(0);
vm->unused_block_warning_table = set_init_numtable();
vm->global_hooks.type = hook_list_type_global;
// setup main thread
th->nt = ZALLOC(struct rb_native_thread);

View File

@ -584,7 +584,7 @@ struct rb_iseq_struct {
} loader;
struct {
struct rb_hook_list_struct *local_hooks;
unsigned int local_hooks_cnt;
rb_event_flag_t global_trace_events;
} exec;
} aux;
@ -650,15 +650,21 @@ void *rb_objspace_alloc(void);
void rb_objspace_free(void *objspace);
void rb_objspace_call_finalizer(void);
enum rb_hook_list_type {
hook_list_type_ractor_local,
hook_list_type_targeted_iseq,
hook_list_type_targeted_def, // C function
hook_list_type_global
};
typedef struct rb_hook_list_struct {
struct rb_event_hook_struct *hooks;
rb_event_flag_t events;
unsigned int running;
enum rb_hook_list_type type;
bool need_clean;
bool is_local;
} rb_hook_list_t;
// see builtin.h for definition
typedef const struct rb_builtin_function *RB_BUILTIN;
@ -2029,8 +2035,9 @@ rb_execution_context_t *rb_vm_main_ractor_ec(rb_vm_t *vm); // ractor.c
RUBY_EXTERN struct rb_ractor_struct *ruby_single_main_ractor; // ractor.c
RUBY_EXTERN rb_vm_t *ruby_current_vm_ptr;
RUBY_EXTERN rb_event_flag_t ruby_vm_event_flags;
RUBY_EXTERN rb_event_flag_t ruby_vm_event_enabled_global_flags;
RUBY_EXTERN unsigned int ruby_vm_event_local_num;
RUBY_EXTERN rb_event_flag_t ruby_vm_event_enabled_global_flags; // only ever added to
RUBY_EXTERN unsigned int ruby_vm_iseq_events_enabled;
RUBY_EXTERN unsigned int ruby_vm_c_events_enabled;
#define GET_VM() rb_current_vm()
#define GET_RACTOR() rb_current_ractor()
@ -2272,8 +2279,9 @@ struct rb_trace_arg_struct {
void rb_hook_list_mark(rb_hook_list_t *hooks);
void rb_hook_list_mark_and_move(rb_hook_list_t *hooks);
void rb_hook_list_free(rb_hook_list_t *hooks);
void rb_hook_list_connect_tracepoint(VALUE target, rb_hook_list_t *list, VALUE tpval, unsigned int target_line);
void rb_hook_list_remove_tracepoint(rb_hook_list_t *list, VALUE tpval);
void rb_hook_list_connect_local_tracepoint(rb_hook_list_t *list, VALUE tpval, unsigned int target_line);
bool rb_hook_list_remove_local_tracepoint(rb_hook_list_t *list, VALUE tpval);
unsigned int rb_hook_list_count(rb_hook_list_t *list);
void rb_exec_event_hooks(struct rb_trace_arg_struct *trace_arg, rb_hook_list_t *hooks, int pop_p);
@ -2312,6 +2320,8 @@ struct rb_ractor_pub {
VALUE self;
uint32_t id;
rb_hook_list_t hooks;
st_table *targeted_hooks; // also called "local hooks". {ISEQ => hook_list, def => hook_list...}
unsigned int targeted_hooks_cnt; // ex: tp.enabled(target: method(:puts))
};
static inline rb_hook_list_t *

View File

@ -3224,8 +3224,7 @@ vm_callee_setup_arg(rb_execution_context_t *ec, struct rb_calling_info *calling,
VM_ASSERT(cc == calling->cc);
if (vm_call_iseq_optimizable_p(ci, cc)) {
if ((iseq->body->builtin_attrs & BUILTIN_ATTR_SINGLE_NOARG_LEAF) &&
!(ruby_vm_event_flags & (RUBY_EVENT_C_CALL | RUBY_EVENT_C_RETURN))) {
if ((iseq->body->builtin_attrs & BUILTIN_ATTR_SINGLE_NOARG_LEAF) && ruby_vm_c_events_enabled == 0) {
VM_ASSERT(iseq->body->builtin_attrs & BUILTIN_ATTR_LEAF);
vm_cc_bf_set(cc, (void *)iseq->body->iseq_encoded[1]);
CC_SET_FASTPATH(cc, vm_call_single_noarg_leaf_builtin, true);
@ -4809,7 +4808,7 @@ NOINLINE(static VALUE vm_call_optimized(rb_execution_context_t *ec, rb_control_f
const struct rb_callinfo *ci, const struct rb_callcache *cc));
#define VM_CALL_METHOD_ATTR(var, func, nohook) \
if (UNLIKELY(ruby_vm_event_flags & (RUBY_EVENT_C_CALL | RUBY_EVENT_C_RETURN))) { \
if (UNLIKELY(ruby_vm_c_events_enabled > 0)) { \
EXEC_EVENT_HOOK(ec, RUBY_EVENT_C_CALL, calling->recv, vm_cc_cme(cc)->def->original_id, \
vm_ci_mid(ci), vm_cc_cme(cc)->owner, Qundef); \
var = func; \
@ -7193,13 +7192,15 @@ NOINLINE(static void vm_trace(rb_execution_context_t *ec, rb_control_frame_t *re
static inline void
vm_trace_hook(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp, const VALUE *pc,
rb_event_flag_t pc_events, rb_event_flag_t target_event,
rb_hook_list_t *global_hooks, rb_hook_list_t *const *local_hooks_ptr, VALUE val)
rb_hook_list_t *global_hooks, rb_hook_list_t *local_hooks, VALUE val)
{
rb_event_flag_t event = pc_events & target_event;
VALUE self = GET_SELF();
VM_ASSERT(rb_popcount64((uint64_t)event) == 1);
if (local_hooks) local_hooks->running++; // make sure they don't get deleted while global hooks run
if (event & global_hooks->events) {
/* increment PC because source line is calculated with PC-1 */
reg_cfp->pc++;
@ -7208,8 +7209,7 @@ vm_trace_hook(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp, const VAL
reg_cfp->pc--;
}
// Load here since global hook above can add and free local hooks
rb_hook_list_t *local_hooks = *local_hooks_ptr;
if (local_hooks) local_hooks->running--;
if (local_hooks != NULL) {
if (event & local_hooks->events) {
/* increment PC because source line is calculated with PC-1 */
@ -7222,7 +7222,7 @@ vm_trace_hook(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp, const VAL
#define VM_TRACE_HOOK(target_event, val) do { \
if ((pc_events & (target_event)) & enabled_flags) { \
vm_trace_hook(ec, reg_cfp, pc, pc_events, (target_event), global_hooks, local_hooks_ptr, (val)); \
vm_trace_hook(ec, reg_cfp, pc, pc_events, (target_event), global_hooks, local_hooks, (val)); \
} \
} while (0)
@ -7238,22 +7238,28 @@ static void
vm_trace(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp)
{
const VALUE *pc = reg_cfp->pc;
rb_event_flag_t enabled_flags = ruby_vm_event_flags & ISEQ_TRACE_EVENTS;
rb_event_flag_t global_events = enabled_flags;
rb_ractor_t *r = rb_ec_ractor_ptr(ec);
rb_event_flag_t enabled_flags = r->pub.hooks.events & ISEQ_TRACE_EVENTS;
rb_event_flag_t ractor_events = enabled_flags;
if (enabled_flags == 0 && ruby_vm_event_local_num == 0) {
if (enabled_flags == 0 && rb_ractor_targeted_hooks_cnt(r) == 0) {
return;
}
else {
const rb_iseq_t *iseq = reg_cfp->iseq;
VALUE iseq_val = (VALUE)iseq;
size_t pos = pc - ISEQ_BODY(iseq)->iseq_encoded;
rb_event_flag_t pc_events = rb_iseq_event_flags(iseq, pos);
rb_hook_list_t *local_hooks = iseq->aux.exec.local_hooks;
rb_hook_list_t *const *local_hooks_ptr = &iseq->aux.exec.local_hooks;
unsigned int local_hooks_cnt = iseq->aux.exec.local_hooks_cnt;
rb_hook_list_t *local_hooks = NULL;
if (RB_UNLIKELY(local_hooks_cnt > 0)) {
st_data_t val;
if (st_lookup(rb_ractor_targeted_hooks(r), (st_data_t)iseq, &val)) {
local_hooks = (rb_hook_list_t*)val;
}
}
rb_event_flag_t iseq_local_events = local_hooks != NULL ? local_hooks->events : 0;
rb_hook_list_t *bmethod_local_hooks = NULL;
rb_hook_list_t **bmethod_local_hooks_ptr = NULL;
rb_event_flag_t bmethod_local_events = 0;
const bool bmethod_frame = VM_FRAME_BMETHOD_P(reg_cfp);
enabled_flags |= iseq_local_events;
@ -7263,14 +7269,18 @@ vm_trace(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp)
if (bmethod_frame) {
const rb_callable_method_entry_t *me = rb_vm_frame_method_entry(reg_cfp);
VM_ASSERT(me->def->type == VM_METHOD_TYPE_BMETHOD);
bmethod_local_hooks = me->def->body.bmethod.hooks;
bmethod_local_hooks_ptr = &me->def->body.bmethod.hooks;
if (bmethod_local_hooks) {
bmethod_local_events = bmethod_local_hooks->events;
unsigned int bmethod_hooks_cnt = me->def->body.bmethod.local_hooks_cnt;
if (RB_UNLIKELY(bmethod_hooks_cnt > 0)) {
st_data_t val;
if (st_lookup(rb_ractor_targeted_hooks(r), (st_data_t)me->def, &val)) {
bmethod_local_hooks = (rb_hook_list_t*)val;
}
if (bmethod_local_hooks) {
bmethod_local_events = bmethod_local_hooks->events;
}
}
}
if ((pc_events & enabled_flags) == 0 && !bmethod_frame) {
#if 0
/* disable trace */
@ -7291,7 +7301,7 @@ vm_trace(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp)
rb_hook_list_t *global_hooks = rb_ec_ractor_hooks(ec);
/* Note, not considering iseq local events here since the same
* iseq could be used in multiple bmethods. */
rb_event_flag_t bmethod_events = global_events | bmethod_local_events;
rb_event_flag_t bmethod_events = ractor_events | bmethod_local_events;
if (0) {
ruby_debug_printf("vm_trace>>%4d (%4x) - %s:%d %s\n",
@ -7307,7 +7317,7 @@ vm_trace(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp)
/* check traces */
if ((pc_events & RUBY_EVENT_B_CALL) && bmethod_frame && (bmethod_events & RUBY_EVENT_CALL)) {
/* b_call instruction running as a method. Fire call event. */
vm_trace_hook(ec, reg_cfp, pc, RUBY_EVENT_CALL, RUBY_EVENT_CALL, global_hooks, bmethod_local_hooks_ptr, Qundef);
vm_trace_hook(ec, reg_cfp, pc, RUBY_EVENT_CALL, RUBY_EVENT_CALL, global_hooks, bmethod_local_hooks, Qundef);
}
VM_TRACE_HOOK(RUBY_EVENT_CLASS | RUBY_EVENT_CALL | RUBY_EVENT_B_CALL, Qundef);
VM_TRACE_HOOK(RUBY_EVENT_RESCUE, rescue_errinfo(ec, reg_cfp));
@ -7317,15 +7327,8 @@ vm_trace(rb_execution_context_t *ec, rb_control_frame_t *reg_cfp)
VM_TRACE_HOOK(RUBY_EVENT_END | RUBY_EVENT_RETURN | RUBY_EVENT_B_RETURN, TOPN(0));
if ((pc_events & RUBY_EVENT_B_RETURN) && bmethod_frame && (bmethod_events & RUBY_EVENT_RETURN)) {
/* b_return instruction running as a method. Fire return event. */
vm_trace_hook(ec, reg_cfp, pc, RUBY_EVENT_RETURN, RUBY_EVENT_RETURN, global_hooks, bmethod_local_hooks_ptr, TOPN(0));
vm_trace_hook(ec, reg_cfp, pc, RUBY_EVENT_RETURN, RUBY_EVENT_RETURN, global_hooks, bmethod_local_hooks, TOPN(0));
}
// Pin the iseq since `local_hooks_ptr` points inside the iseq's slot on the GC heap.
// We need the pointer to stay valid in case compaction happens in a trace hook.
//
// Similar treatment is unnecessary for `bmethod_local_hooks_ptr` since
// storage for `rb_method_definition_t` is not on the GC heap.
RB_GC_GUARD(iseq_val);
}
}
}

View File

@ -826,7 +826,7 @@ rb_add_method_optimized(VALUE klass, ID mid, enum method_optimized_type opt_type
}
static void
rb_method_definition_release(rb_method_definition_t *def)
method_definition_release(rb_method_definition_t *def)
{
if (def != NULL) {
const unsigned int reference_count_was = RUBY_ATOMIC_FETCH_SUB(def->reference_count, 1);
@ -836,9 +836,6 @@ rb_method_definition_release(rb_method_definition_t *def)
if (reference_count_was == 1) {
if (METHOD_DEBUG) fprintf(stderr, "-%p-%s:1->0 (remove)\n", (void *)def,
rb_id2name(def->original_id));
if (def->type == VM_METHOD_TYPE_BMETHOD && def->body.bmethod.hooks) {
xfree(def->body.bmethod.hooks);
}
xfree(def);
}
else {
@ -848,6 +845,12 @@ rb_method_definition_release(rb_method_definition_t *def)
}
}
void
rb_method_definition_release(rb_method_definition_t *def)
{
method_definition_release(def);
}
static void delete_overloaded_cme(const rb_callable_method_entry_t *cme);
void
@ -872,7 +875,7 @@ rb_free_method_entry(const rb_method_entry_t *me)
// to remove from `Invariants` here.
#endif
rb_method_definition_release(me->def);
method_definition_release(me->def);
}
static inline rb_method_entry_t *search_method(VALUE klass, ID id, VALUE *defined_class_ptr);
@ -939,6 +942,7 @@ setup_method_cfunc_struct(rb_method_cfunc_t *cfunc, VALUE (*func)(ANYARGS), int
cfunc->invoker = call_cfunc_invoker_func(argc);
}
static rb_method_definition_t *
method_definition_addref(rb_method_definition_t *def, bool complemented)
{
@ -952,10 +956,16 @@ method_definition_addref(rb_method_definition_t *def, bool complemented)
return def;
}
void
rb_method_definition_addref(rb_method_definition_t *def)
{
method_definition_addref(def, false);
}
void
rb_method_definition_set(const rb_method_entry_t *me, rb_method_definition_t *def, void *opts)
{
rb_method_definition_release(me->def);
method_definition_release(me->def);
*(rb_method_definition_t **)&me->def = method_definition_addref(def, METHOD_ENTRY_COMPLEMENTED(me));
if (!ruby_running) add_opt_method_entry(me);
@ -1060,8 +1070,6 @@ method_definition_reset(const rb_method_entry_t *me)
break;
case VM_METHOD_TYPE_BMETHOD:
RB_OBJ_WRITTEN(me, Qundef, def->body.bmethod.proc);
/* give up to check all in a list */
if (def->body.bmethod.hooks) rb_gc_writebarrier_remember((VALUE)me);
break;
case VM_METHOD_TYPE_REFINED:
RB_OBJ_WRITTEN(me, Qundef, def->body.refined.orig_me);
@ -1195,7 +1203,7 @@ rb_method_entry_complement_defined_class(const rb_method_entry_t *src_me, ID cal
void
rb_method_entry_copy(rb_method_entry_t *dst, const rb_method_entry_t *src)
{
rb_method_definition_release(dst->def);
method_definition_release(dst->def);
*(rb_method_definition_t **)&dst->def = method_definition_addref(src->def, METHOD_ENTRY_COMPLEMENTED(src));
method_definition_reset(dst);
dst->called_id = src->called_id;

View File

@ -34,6 +34,7 @@
#include "ruby/debug.h"
#include "vm_core.h"
#include "ruby/ractor.h"
#include "ractor_core.h"
#include "yjit.h"
#include "zjit.h"
@ -103,51 +104,90 @@ rb_hook_list_free(rb_hook_list_t *hooks)
void rb_clear_attr_ccs(void);
void rb_clear_bf_ccs(void);
static void
update_global_event_hook(rb_event_flag_t prev_events, rb_event_flag_t new_events)
static bool iseq_trace_set_all_needed(rb_event_flag_t new_events)
{
rb_event_flag_t new_iseq_events = new_events & ISEQ_TRACE_EVENTS;
rb_event_flag_t enabled_iseq_events = ruby_vm_event_enabled_global_flags & ISEQ_TRACE_EVENTS;
bool first_time_iseq_events_p = new_iseq_events & ~enabled_iseq_events;
bool enable_c_call = (prev_events & RUBY_EVENT_C_CALL) == 0 && (new_events & RUBY_EVENT_C_CALL);
return new_iseq_events & ~enabled_iseq_events;
}
static bool clear_attr_ccs_needed(rb_event_flag_t prev_events, rb_event_flag_t new_events)
{
bool enable_c_call = (prev_events & RUBY_EVENT_C_CALL) == 0 && (new_events & RUBY_EVENT_C_CALL);
bool enable_c_return = (prev_events & RUBY_EVENT_C_RETURN) == 0 && (new_events & RUBY_EVENT_C_RETURN);
bool enable_call = (prev_events & RUBY_EVENT_CALL) == 0 && (new_events & RUBY_EVENT_CALL);
bool enable_return = (prev_events & RUBY_EVENT_RETURN) == 0 && (new_events & RUBY_EVENT_RETURN);
return enable_c_call || enable_c_return;
}
/* If the events are internal events (e.g. gc hooks), it updates them globally for all ractors. Otherwise
* they are ractor local. You cannot listen to internal events through set_trace_func or TracePoint.
* Some ractor-local tracepoint events cause global level iseq changes, so are still called `global events`.
*/
static void
update_global_event_hooks(rb_hook_list_t *list, rb_event_flag_t prev_events, rb_event_flag_t new_events, int change_iseq_events, int change_c_events)
{
rb_execution_context_t *ec = rb_current_execution_context(false);
unsigned int lev;
// Can't enter VM lock during freeing of ractor hook list on MMTK, where ec == NULL.
if (ec) {
RB_VM_LOCK_ENTER_LEV(&lev);
rb_vm_barrier();
}
rb_event_flag_t new_iseq_events = new_events & ISEQ_TRACE_EVENTS;
rb_event_flag_t enabled_iseq_events = ruby_vm_event_enabled_global_flags & ISEQ_TRACE_EVENTS;
bool new_iseq_events_p = iseq_trace_set_all_needed(new_events);
bool enable_call = (prev_events & RUBY_EVENT_CALL) == 0 && (new_events & RUBY_EVENT_CALL);
bool enable_return = (prev_events & RUBY_EVENT_RETURN) == 0 && (new_events & RUBY_EVENT_RETURN);
bool clear_attr_ccs_p = clear_attr_ccs_needed(prev_events, new_events);
// FIXME: `ruby_vm_event_flags` should have the global list of event flags for internal events as well
// as for all ractors. That's not how it works right now, so we shouldn't rely on it apart from the
// internal events. Since it doesn't work like this, we have to track more state with `ruby_vm_iseq_events_enabled`,
// `ruby_vm_c_events_enabled`, etc.
rb_event_flag_t new_events_global = (ruby_vm_event_flags & ~prev_events) | new_events;
ruby_vm_event_flags = new_events_global;
// Modify ISEQs or CCs to enable tracing
if (first_time_iseq_events_p) {
if (new_iseq_events_p) {
// write all ISeqs only when new events are added for the first time
rb_iseq_trace_set_all(new_iseq_events | enabled_iseq_events);
}
// if c_call or c_return is activated
else if (enable_c_call || enable_c_return) {
else if (clear_attr_ccs_p) { // turn on C_CALL or C_RETURN ractor locally
rb_clear_attr_ccs();
}
else if (enable_call || enable_return) {
else if (enable_call || enable_return) { // turn on CALL or RETURN ractor locally
rb_clear_bf_ccs();
}
// FIXME: Which flags are enabled globally comes from multiple lists, one
// per-ractor and a global list.
// This incorrectly assumes the lists have mutually exclusive flags set.
// This is true for the global (objspace) events, but not for ex. multiple
// Ractors listening for the same iseq events.
rb_event_flag_t new_events_global = (ruby_vm_event_flags & ~prev_events) | new_events;
ruby_vm_event_flags = new_events_global;
ruby_vm_event_enabled_global_flags |= new_events_global;
rb_objspace_set_event_hook(new_events_global);
if (change_iseq_events < 0) {
RUBY_ASSERT(ruby_vm_iseq_events_enabled >= (unsigned int)(-change_iseq_events));
}
ruby_vm_iseq_events_enabled += change_iseq_events;
if (change_c_events < 0) {
RUBY_ASSERT(ruby_vm_c_events_enabled >= (unsigned int)(-change_iseq_events));
}
ruby_vm_c_events_enabled += change_c_events;
ruby_vm_event_enabled_global_flags |= new_events; // NOTE: this is only ever added to
if (new_events_global & RUBY_INTERNAL_EVENT_MASK) {
rb_objspace_set_event_hook(new_events_global);
}
// Invalidate JIT code as needed
if (first_time_iseq_events_p || enable_c_call || enable_c_return) {
if (new_iseq_events_p || clear_attr_ccs_p) {
// Invalidate all code when ISEQs are modified to use trace_* insns above.
// Also invalidate when enabling c_call or c_return because generated code
// never fires these events.
// Internal events fire inside C routines so don't need special handling.
// Do this after event flags updates so other ractors see updated vm events
// when they wake up.
rb_yjit_tracing_invalidate_all();
rb_zjit_tracing_invalidate_all();
}
if (ec) {
RB_VM_LOCK_LEAVE_LEV(&lev);
}
}
/* add/remove hooks */
@ -174,25 +214,30 @@ alloc_event_hook(rb_event_hook_func_t func, rb_event_flag_t events, VALUE data,
return hook;
}
// Connect a hook onto a ractor, an iseq or a method definition's hook list
static void
hook_list_connect(VALUE list_owner, rb_hook_list_t *list, rb_event_hook_t *hook, int global_p)
hook_list_connect(rb_hook_list_t *list, rb_event_hook_t *hook, int global_p)
{
rb_event_flag_t prev_events = list->events;
int change_iseq_events = 0;
int change_c_events = 0;
hook->next = list->hooks;
list->hooks = hook;
list->events |= hook->events;
if (global_p) {
/* global hooks are root objects at GC mark. */
update_global_event_hook(prev_events, list->events);
}
else {
RB_OBJ_WRITTEN(list_owner, Qundef, hook->data);
if (hook->events & ISEQ_TRACE_EVENTS) {
change_iseq_events++;
}
if ((hook->events & RUBY_EVENT_C_CALL) || (hook->events & RUBY_EVENT_C_RETURN)) {
change_c_events++;
}
update_global_event_hooks(list, prev_events, list->events, change_iseq_events, change_c_events);
}
}
static void
connect_event_hook(const rb_execution_context_t *ec, rb_event_hook_t *hook)
connect_non_targeted_event_hook(const rb_execution_context_t *ec, rb_event_hook_t *hook)
{
rb_hook_list_t *list;
@ -205,7 +250,7 @@ connect_event_hook(const rb_execution_context_t *ec, rb_event_hook_t *hook)
else {
list = rb_ec_ractor_hooks(ec);
}
hook_list_connect(Qundef, list, hook, TRUE);
hook_list_connect(list, hook, TRUE);
}
static void
@ -214,7 +259,7 @@ rb_threadptr_add_event_hook(const rb_execution_context_t *ec, rb_thread_t *th,
{
rb_event_hook_t *hook = alloc_event_hook(func, events, data, hook_flags);
hook->filter.th = th;
connect_event_hook(ec, hook);
connect_non_targeted_event_hook(ec, hook);
}
void
@ -239,7 +284,35 @@ void
rb_add_event_hook2(rb_event_hook_func_t func, rb_event_flag_t events, VALUE data, rb_event_hook_flag_t hook_flags)
{
rb_event_hook_t *hook = alloc_event_hook(func, events, data, hook_flags);
connect_event_hook(GET_EC(), hook);
connect_non_targeted_event_hook(GET_EC(), hook);
}
static bool
hook_list_targeted_p(rb_hook_list_t *list)
{
switch (list->type) {
case hook_list_type_targeted_iseq:
case hook_list_type_targeted_def:
return true;
default:
return false;
}
}
unsigned int
rb_hook_list_count(rb_hook_list_t *list)
{
rb_event_hook_t *hook = list->hooks;
unsigned int count = 0;
while (hook) {
if (!(hook->hook_flags & RUBY_EVENT_HOOK_FLAG_DELETED)) {
count++;
}
hook = hook->next;
}
return count;
}
static void
@ -247,6 +320,8 @@ clean_hooks(rb_hook_list_t *list)
{
rb_event_hook_t *hook, **nextp = &list->hooks;
rb_event_flag_t prev_events = list->events;
int change_iseq_events = 0;
int change_c_events = 0;
VM_ASSERT(list->running == 0);
VM_ASSERT(list->need_clean == true);
@ -257,6 +332,14 @@ clean_hooks(rb_hook_list_t *list)
while ((hook = *nextp) != 0) {
if (hook->hook_flags & RUBY_EVENT_HOOK_FLAG_DELETED) {
*nextp = hook->next;
if (!hook_list_targeted_p(list)) {
if (hook->events & ISEQ_TRACE_EVENTS) {
change_iseq_events--;
}
if ((hook->events & RUBY_EVENT_C_CALL) || (hook->events & RUBY_EVENT_C_RETURN)) {
change_c_events--;
}
}
xfree(hook);
}
else {
@ -265,14 +348,13 @@ clean_hooks(rb_hook_list_t *list)
}
}
if (list->is_local) {
if (hook_list_targeted_p(list)) {
if (list->events == 0) {
/* local events */
ruby_xfree(list);
}
}
else {
update_global_event_hook(prev_events, list->events);
update_global_event_hooks(list, prev_events, list->events, change_iseq_events, change_c_events);
}
}
@ -299,8 +381,8 @@ remove_event_hook_from_list(rb_hook_list_t *list, const rb_thread_t *filter_th,
if (hook->filter.th == filter_th || filter_th == MATCH_ANY_FILTER_TH) {
if (UNDEF_P(data) || hook->data == data) {
hook->hook_flags |= RUBY_EVENT_HOOK_FLAG_DELETED;
ret+=1;
list->need_clean = true;
ret+=1;
}
}
}
@ -1217,7 +1299,7 @@ tp_call_trace(VALUE tpval, rb_trace_arg_t *trace_arg)
(*tp->func)(tpval, tp->data);
}
else {
if (tp->ractor == NULL || tp->ractor == GET_RACTOR()) {
if (tp->ractor == GET_RACTOR()) {
rb_proc_call_with_block((VALUE)tp->proc, 1, &tpval, Qnil);
}
}
@ -1263,14 +1345,33 @@ iseq_of(VALUE target)
const rb_method_definition_t *rb_method_def(VALUE method); /* proc.c */
rb_hook_list_t *
rb_method_def_local_hooks(rb_method_definition_t *def, rb_ractor_t *cr, bool create)
{
st_data_t val;
rb_hook_list_t *hook_list = NULL;
if (st_lookup(rb_ractor_targeted_hooks(cr), (st_data_t)def, &val)) {
hook_list = (rb_hook_list_t*)val;
RUBY_ASSERT(hook_list->type == hook_list_type_targeted_def);
}
else if (create) {
hook_list = ZALLOC(rb_hook_list_t);
hook_list->type = hook_list_type_targeted_def;
st_insert(cr->pub.targeted_hooks, (st_data_t)def, (st_data_t)hook_list);
}
return hook_list;
}
// Enable "local" (targeted) tracepoint
static VALUE
rb_tracepoint_enable_for_target(VALUE tpval, VALUE target, VALUE target_line)
{
rb_tp_t *tp = tpptr(tpval);
const rb_iseq_t *iseq = iseq_of(target);
const rb_iseq_t *iseq = iseq_of(target); // takes Proc, Iseq, Method
int n = 0;
unsigned int line = 0;
bool target_bmethod = false;
rb_ractor_t *cr = GET_RACTOR();
if (tp->tracing > 0) {
rb_raise(rb_eArgError, "can't nest-enable a targeting TracePoint");
@ -1288,62 +1389,80 @@ rb_tracepoint_enable_for_target(VALUE tpval, VALUE target, VALUE target_line)
VM_ASSERT(tp->local_target_set == Qfalse);
RB_OBJ_WRITE(tpval, &tp->local_target_set, rb_obj_hide(rb_ident_hash_new()));
/* bmethod */
if (rb_obj_is_method(target)) {
rb_method_definition_t *def = (rb_method_definition_t *)rb_method_def(target);
if (def->type == VM_METHOD_TYPE_BMETHOD &&
(tp->events & (RUBY_EVENT_CALL | RUBY_EVENT_RETURN))) {
if (def->body.bmethod.hooks == NULL) {
def->body.bmethod.hooks = ZALLOC(rb_hook_list_t);
def->body.bmethod.hooks->is_local = true;
RB_VM_LOCKING() {
// Rewriting iseq instructions across ractors is not safe unless they are stopped.
rb_vm_barrier();
/* bmethod */
if (rb_obj_is_method(target)) {
rb_method_definition_t *def = (rb_method_definition_t *)rb_method_def(target);
if (def->type == VM_METHOD_TYPE_BMETHOD && (tp->events & (RUBY_EVENT_CALL | RUBY_EVENT_RETURN))) {
rb_hook_list_t *hook_list = rb_method_def_local_hooks(def, cr, true);
rb_hook_list_connect_local_tracepoint(hook_list, tpval, 0);
rb_hash_aset(tp->local_target_set, target, Qfalse); // Qfalse means not an iseq
rb_method_definition_addref(def); // in case `tp` gets GC'd and didn't disable the hook, `def` needs to stay alive
def->body.bmethod.local_hooks_cnt++;
target_bmethod = true;
n++;
}
rb_hook_list_connect_tracepoint(target, def->body.bmethod.hooks, tpval, 0);
rb_hash_aset(tp->local_target_set, target, Qfalse);
target_bmethod = true;
n++;
}
}
/* iseq */
n += rb_iseq_add_local_tracepoint_recursively(iseq, tp->events, tpval, line, target_bmethod);
rb_hash_aset(tp->local_target_set, (VALUE)iseq, Qtrue);
/* iseq */
n += rb_iseq_add_local_tracepoint_recursively(iseq, tp->events, tpval, line, target_bmethod);
if (n > 0) {
rb_hash_aset(tp->local_target_set, (VALUE)iseq, Qtrue);
if ((tp->events & (RUBY_EVENT_CALL | RUBY_EVENT_RETURN)) &&
iseq->body->builtin_attrs & BUILTIN_ATTR_SINGLE_NOARG_LEAF) {
rb_clear_bf_ccs();
if ((tp->events & (RUBY_EVENT_CALL | RUBY_EVENT_RETURN)) &&
iseq->body->builtin_attrs & BUILTIN_ATTR_SINGLE_NOARG_LEAF) {
rb_clear_bf_ccs();
}
rb_yjit_tracing_invalidate_all();
rb_zjit_tracing_invalidate_all();
rb_ractor_targeted_hooks_incr(tp->ractor);
if (tp->events & ISEQ_TRACE_EVENTS) {
ruby_vm_iseq_events_enabled++;
}
if ((tp->events & RUBY_EVENT_C_CALL) || (tp->events & RUBY_EVENT_C_RETURN)) {
ruby_vm_c_events_enabled++;
}
tp->tracing = 1;
}
}
if (n == 0) {
rb_raise(rb_eArgError, "can not enable any hooks");
}
rb_yjit_tracing_invalidate_all();
rb_zjit_tracing_invalidate_all();
ruby_vm_event_local_num++;
tp->tracing = 1;
return Qnil;
}
static int
disable_local_event_iseq_i(VALUE target, VALUE iseq_p, VALUE tpval)
disable_local_tracepoint_i(VALUE target, VALUE iseq_p, VALUE tpval)
{
rb_tp_t *tp = tpptr(tpval);
rb_ractor_t *cr;
rb_method_definition_t *def;
rb_hook_list_t *hook_list;
ASSERT_vm_locking_with_barrier();
if (iseq_p) {
rb_iseq_remove_local_tracepoint_recursively((rb_iseq_t *)target, tpval);
rb_iseq_remove_local_tracepoint_recursively((rb_iseq_t *)target, tpval, tp->ractor);
}
else {
cr = GET_RACTOR();
/* bmethod */
rb_method_definition_t *def = (rb_method_definition_t *)rb_method_def(target);
rb_hook_list_t *hooks = def->body.bmethod.hooks;
VM_ASSERT(hooks != NULL);
rb_hook_list_remove_tracepoint(hooks, tpval);
if (hooks->events == 0) {
rb_hook_list_free(def->body.bmethod.hooks);
def->body.bmethod.hooks = NULL;
def = (rb_method_definition_t *)rb_method_def(target);
hook_list = rb_method_def_local_hooks(def, cr, false);
RUBY_ASSERT(hook_list != NULL);
if (rb_hook_list_remove_local_tracepoint(hook_list, tpval)) {
RUBY_ASSERT(def->body.bmethod.local_hooks_cnt > 0);
def->body.bmethod.local_hooks_cnt--;
if (hook_list->events == 0) {
st_delete(rb_ractor_targeted_hooks(cr), (st_data_t*)&def, NULL);
rb_hook_list_free(hook_list);
}
rb_method_definition_release(def);
}
}
return ST_CONTINUE;
@ -1356,10 +1475,23 @@ rb_tracepoint_disable(VALUE tpval)
tp = tpptr(tpval);
if (tp->local_target_set) {
rb_hash_foreach(tp->local_target_set, disable_local_event_iseq_i, tpval);
RB_OBJ_WRITE(tpval, &tp->local_target_set, Qfalse);
ruby_vm_event_local_num--;
if (RTEST(tp->local_target_set)) {
RUBY_ASSERT(GET_RACTOR() == tp->ractor);
RB_VM_LOCKING() {
rb_vm_barrier();
rb_hash_foreach(tp->local_target_set, disable_local_tracepoint_i, tpval);
RB_OBJ_WRITE(tpval, &tp->local_target_set, Qfalse);
rb_ractor_targeted_hooks_decr(tp->ractor);
if (tp->events & ISEQ_TRACE_EVENTS) {
RUBY_ASSERT(ruby_vm_iseq_events_enabled > 0);
ruby_vm_iseq_events_enabled--;
}
if ((tp->events & RUBY_EVENT_C_CALL) || (tp->events & RUBY_EVENT_C_RETURN)) {
RUBY_ASSERT(ruby_vm_c_events_enabled > 0);
ruby_vm_c_events_enabled--;
}
}
}
else {
if (tp->target_th) {
@ -1374,26 +1506,30 @@ rb_tracepoint_disable(VALUE tpval)
return Qundef;
}
// connect a targeted (ie: "local") tracepoint to the hook list for the method
// ex: tp.enable(target: method(:puts))
void
rb_hook_list_connect_tracepoint(VALUE target, rb_hook_list_t *list, VALUE tpval, unsigned int target_line)
rb_hook_list_connect_local_tracepoint(rb_hook_list_t *list, VALUE tpval, unsigned int target_line)
{
rb_tp_t *tp = tpptr(tpval);
rb_event_hook_t *hook = alloc_event_hook((rb_event_hook_func_t)tp_call_trace, tp->events & ISEQ_TRACE_EVENTS, tpval,
RUBY_EVENT_HOOK_FLAG_SAFE | RUBY_EVENT_HOOK_FLAG_RAW_ARG);
hook->filter.target_line = target_line;
hook_list_connect(target, list, hook, FALSE);
hook_list_connect(list, hook, FALSE);
}
void
rb_hook_list_remove_tracepoint(rb_hook_list_t *list, VALUE tpval)
bool
rb_hook_list_remove_local_tracepoint(rb_hook_list_t *list, VALUE tpval)
{
rb_event_hook_t *hook = list->hooks;
rb_event_flag_t events = 0;
bool removed = false;
while (hook) {
if (hook->data == tpval) {
hook->hook_flags |= RUBY_EVENT_HOOK_FLAG_DELETED;
list->need_clean = true;
removed = true;
}
else if ((hook->hook_flags & RUBY_EVENT_HOOK_FLAG_DELETED) == 0) {
events |= hook->events;
@ -1402,6 +1538,7 @@ rb_hook_list_remove_tracepoint(rb_hook_list_t *list, VALUE tpval)
}
list->events = events;
return removed;
}
static VALUE
@ -1496,8 +1633,8 @@ tracepoint_new(VALUE klass, rb_thread_t *target_th, rb_event_flag_t events, void
TypedData_Get_Struct(tpval, rb_tp_t, &tp_data_type, tp);
RB_OBJ_WRITE(tpval, &tp->proc, proc);
tp->ractor = rb_ractor_shareable_p(proc) ? NULL : GET_RACTOR();
tp->func = func;
tp->ractor = GET_RACTOR();
tp->func = func; // for internal events
tp->data = data;
tp->events = events;
tp->self = tpval;
@ -1512,9 +1649,6 @@ rb_tracepoint_new(VALUE target_thval, rb_event_flag_t events, void (*func)(VALUE
if (RTEST(target_thval)) {
target_th = rb_thread_ptr(target_thval);
/* TODO: Test it!
* Warning: This function is not tested.
*/
}
return tracepoint_new(rb_cTracePoint, target_th, events, func, data, Qundef);
}
@ -1621,7 +1755,6 @@ tracepoint_stat_s(rb_execution_context_t *ec, VALUE self)
VALUE stat = rb_hash_new();
tracepoint_stat_event_hooks(stat, vm->self, rb_ec_ractor_hooks(ec)->hooks);
/* TODO: thread local hooks */
return stat;
}

13
yjit.c
View File

@ -160,18 +160,7 @@ rb_yjit_exit_locations_dict(VALUE *yjit_raw_samples, int *yjit_line_samples, int
bool
rb_c_method_tracing_currently_enabled(const rb_execution_context_t *ec)
{
rb_event_flag_t tracing_events;
if (rb_multi_ractor_p()) {
tracing_events = ruby_vm_event_enabled_global_flags;
}
else {
// At the time of writing, events are never removed from
// ruby_vm_event_enabled_global_flags so always checking using it would
// mean we don't compile even after tracing is disabled.
tracing_events = rb_ec_ractor_hooks(ec)->events;
}
return tracing_events & (RUBY_EVENT_C_CALL | RUBY_EVENT_C_RETURN);
return ruby_vm_c_events_enabled > 0;
}
// The code we generate in gen_send_cfunc() doesn't fire the c_return TracePoint event

View File

@ -1214,19 +1214,10 @@ pub struct rb_iseq_struct__bindgen_ty_1__bindgen_ty_1 {
#[repr(C)]
#[derive(Debug, Copy, Clone)]
pub struct rb_iseq_struct__bindgen_ty_1__bindgen_ty_2 {
pub local_hooks: *mut rb_hook_list_struct,
pub local_hooks_cnt: ::std::os::raw::c_uint,
pub global_trace_events: rb_event_flag_t,
}
#[repr(C)]
#[derive(Debug, Copy, Clone)]
pub struct rb_hook_list_struct {
pub hooks: *mut rb_event_hook_struct,
pub events: rb_event_flag_t,
pub running: ::std::os::raw::c_uint,
pub need_clean: bool,
pub is_local: bool,
}
#[repr(C)]
pub struct rb_captured_block {
pub self_: VALUE,
pub ep: *const VALUE,
@ -1846,11 +1837,6 @@ pub type rb_iseq_param_keyword_struct =
pub struct succ_index_table {
pub _address: u8,
}
#[repr(C)]
#[derive(Debug, Copy, Clone)]
pub struct rb_event_hook_struct {
pub _address: u8,
}
unsafe extern "C" {
pub fn ruby_xfree(ptr: *mut ::std::os::raw::c_void);
pub fn rb_class_attached_object(klass: VALUE) -> VALUE;