зеркало из https://github.com/github/ruby.git
Lazily move units from active_units to stale_units
to avoid SEGV like http://ci.rvm.jp/results/trunk-mjit@phosphorus-docker/3289588 by a race condition between mjit_recompile and compation around active_units
This commit is contained in:
Родитель
0a52161872
Коммит
5d8f227d0e
7
mjit.c
7
mjit.c
|
@ -333,12 +333,13 @@ mjit_recompile(const rb_iseq_t *iseq)
|
||||||
verbose(1, "JIT recompile: %s@%s:%d", RSTRING_PTR(iseq->body->location.label),
|
verbose(1, "JIT recompile: %s@%s:%d", RSTRING_PTR(iseq->body->location.label),
|
||||||
RSTRING_PTR(rb_iseq_path(iseq)), FIX2INT(iseq->body->location.first_lineno));
|
RSTRING_PTR(rb_iseq_path(iseq)), FIX2INT(iseq->body->location.first_lineno));
|
||||||
|
|
||||||
|
// Lazily move active_units to stale_units to avoid race conditions around active_units with compaction
|
||||||
CRITICAL_SECTION_START(3, "in rb_mjit_recompile_iseq");
|
CRITICAL_SECTION_START(3, "in rb_mjit_recompile_iseq");
|
||||||
remove_from_list(iseq->body->jit_unit, &active_units);
|
iseq->body->jit_unit->stale_p = true;
|
||||||
iseq->body->jit_func = (mjit_func_t)NOT_ADDED_JIT_ISEQ_FUNC;
|
pending_stale_p = true;
|
||||||
add_to_list(iseq->body->jit_unit, &stale_units);
|
|
||||||
CRITICAL_SECTION_FINISH(3, "in rb_mjit_recompile_iseq");
|
CRITICAL_SECTION_FINISH(3, "in rb_mjit_recompile_iseq");
|
||||||
|
|
||||||
|
iseq->body->jit_func = (mjit_func_t)NOT_ADDED_JIT_ISEQ_FUNC;
|
||||||
mjit_add_iseq_to_process(iseq, &iseq->body->jit_unit->compile_info);
|
mjit_add_iseq_to_process(iseq, &iseq->body->jit_unit->compile_info);
|
||||||
if (UNLIKELY(mjit_opts.wait)) {
|
if (UNLIKELY(mjit_opts.wait)) {
|
||||||
mjit_wait(iseq->body);
|
mjit_wait(iseq->body);
|
||||||
|
|
|
@ -160,6 +160,8 @@ struct rb_mjit_unit {
|
||||||
#endif
|
#endif
|
||||||
// Only used by unload_units. Flag to check this unit is currently on stack or not.
|
// Only used by unload_units. Flag to check this unit is currently on stack or not.
|
||||||
bool used_code_p;
|
bool used_code_p;
|
||||||
|
// True if this is still in active_units but it's to be lazily removed
|
||||||
|
bool stale_p;
|
||||||
// mjit_compile's optimization switches
|
// mjit_compile's optimization switches
|
||||||
struct rb_mjit_compile_info compile_info;
|
struct rb_mjit_compile_info compile_info;
|
||||||
// captured CC values, they should be marked with iseq.
|
// captured CC values, they should be marked with iseq.
|
||||||
|
@ -225,6 +227,8 @@ static rb_nativethread_cond_t mjit_gc_wakeup;
|
||||||
static int in_gc = 0;
|
static int in_gc = 0;
|
||||||
// True when JIT is working.
|
// True when JIT is working.
|
||||||
static bool in_jit = false;
|
static bool in_jit = false;
|
||||||
|
// True when active_units has at least one stale_p=true unit.
|
||||||
|
static bool pending_stale_p = false;
|
||||||
// The times when unload_units is requested. unload_units is called after some requests.
|
// The times when unload_units is requested. unload_units is called after some requests.
|
||||||
static int unload_requests = 0;
|
static int unload_requests = 0;
|
||||||
// The total number of unloaded units.
|
// The total number of unloaded units.
|
||||||
|
@ -946,9 +950,7 @@ compile_compact_jit_code(char* c_file)
|
||||||
// TODO: Consider using a more granular lock after we implement inlining across
|
// TODO: Consider using a more granular lock after we implement inlining across
|
||||||
// compacted functions (not done yet).
|
// compacted functions (not done yet).
|
||||||
bool success = true;
|
bool success = true;
|
||||||
CRITICAL_SECTION_START(3, "before active_units list_for_each");
|
list_for_each(&active_units.head, child_unit, unode) {
|
||||||
list_for_each_safe(&active_units.head, child_unit, next, unode) {
|
|
||||||
CRITICAL_SECTION_FINISH(3, "after active_units list_for_each");
|
|
||||||
char funcname[MAXPATHLEN];
|
char funcname[MAXPATHLEN];
|
||||||
sprint_funcname(funcname, child_unit);
|
sprint_funcname(funcname, child_unit);
|
||||||
|
|
||||||
|
@ -962,10 +964,7 @@ compile_compact_jit_code(char* c_file)
|
||||||
if (!iseq_label) iseq_label = sep = "";
|
if (!iseq_label) iseq_label = sep = "";
|
||||||
fprintf(f, "\n/* %s%s%s:%ld */\n", iseq_label, sep, iseq_path, iseq_lineno);
|
fprintf(f, "\n/* %s%s%s:%ld */\n", iseq_label, sep, iseq_path, iseq_lineno);
|
||||||
success &= mjit_compile(f, child_unit->iseq, funcname, child_unit->id);
|
success &= mjit_compile(f, child_unit->iseq, funcname, child_unit->id);
|
||||||
|
|
||||||
CRITICAL_SECTION_START(3, "before active_units list_for_each");
|
|
||||||
}
|
}
|
||||||
CRITICAL_SECTION_FINISH(3, "after active_units list_for_each");
|
|
||||||
|
|
||||||
// release blocking mjit_gc_start_hook
|
// release blocking mjit_gc_start_hook
|
||||||
CRITICAL_SECTION_START(3, "after mjit_compile to wakeup client for GC");
|
CRITICAL_SECTION_START(3, "after mjit_compile to wakeup client for GC");
|
||||||
|
@ -1376,10 +1375,23 @@ mjit_worker(void)
|
||||||
|
|
||||||
// Wait until a unit becomes available
|
// Wait until a unit becomes available
|
||||||
CRITICAL_SECTION_START(3, "in worker dequeue");
|
CRITICAL_SECTION_START(3, "in worker dequeue");
|
||||||
while ((list_empty(&unit_queue.head) || active_units.length >= mjit_opts.max_cache_size) && !stop_worker_p) {
|
while ((pending_stale_p || list_empty(&unit_queue.head) || active_units.length >= mjit_opts.max_cache_size) && !stop_worker_p) {
|
||||||
rb_native_cond_wait(&mjit_worker_wakeup, &mjit_engine_mutex);
|
rb_native_cond_wait(&mjit_worker_wakeup, &mjit_engine_mutex);
|
||||||
verbose(3, "Getting wakeup from client");
|
verbose(3, "Getting wakeup from client");
|
||||||
|
|
||||||
|
// Lazily move active_units to stale_units to avoid race conditions around active_units with compaction
|
||||||
|
if (pending_stale_p) {
|
||||||
|
pending_stale_p = false;
|
||||||
|
struct rb_mjit_unit *next;
|
||||||
|
list_for_each_safe(&active_units.head, unit, next, unode) {
|
||||||
|
if (unit->stale_p) {
|
||||||
|
unit->stale_p = false;
|
||||||
|
remove_from_list(unit, &active_units);
|
||||||
|
add_to_list(unit, &stale_units);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Unload some units as needed
|
// Unload some units as needed
|
||||||
if (unload_requests >= throttle_threshold) {
|
if (unload_requests >= throttle_threshold) {
|
||||||
while (in_gc) {
|
while (in_gc) {
|
||||||
|
|
Загрузка…
Ссылка в новой задаче