thread_pthread.c (gvl_acquire_common): persist timeout across calls
Reuse old expiration time if the previous native_cond_timedwait did not return ETIMEDOUT. This should improve timeslice accuracy for Timeout.timeout rubyspec without causing excessive wakeups on uncontended GVL acquisition. cf. http://ci.rvm.jp/results/trunk-gc-asserts@silicon-docker/1180486 http://ci.rvm.jp/results/trunk-gc-asserts@silicon-docker/1184623 git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@64165 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
0a03394b44
commit
8c2ae6e3ed
@ -95,14 +95,20 @@ gvl_acquire_common(rb_vm_t *vm, rb_thread_t *th)
|
|||||||
list_add_tail(&vm->gvl.waitq, &nd->ubf_list);
|
list_add_tail(&vm->gvl.waitq, &nd->ubf_list);
|
||||||
do {
|
do {
|
||||||
if (!vm->gvl.timer) {
|
if (!vm->gvl.timer) {
|
||||||
struct timespec ts = { 0, TIME_QUANTUM_USEC * 1000 };
|
static struct timespec ts;
|
||||||
|
static int err = ETIMEDOUT;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* become designated timer thread to kick vm->gvl.acquired
|
* become designated timer thread to kick vm->gvl.acquired
|
||||||
* periodically
|
* periodically. Continue on old timeout if it expired:
|
||||||
*/
|
*/
|
||||||
|
if (err == ETIMEDOUT) {
|
||||||
|
ts.tv_sec = 0;
|
||||||
|
ts.tv_nsec = TIME_QUANTUM_USEC * 1000;
|
||||||
ts = native_cond_timeout(&nd->cond.gvlq, ts);
|
ts = native_cond_timeout(&nd->cond.gvlq, ts);
|
||||||
|
}
|
||||||
vm->gvl.timer = th;
|
vm->gvl.timer = th;
|
||||||
native_cond_timedwait(&nd->cond.gvlq, &vm->gvl.lock, &ts);
|
err = native_cond_timedwait(&nd->cond.gvlq, &vm->gvl.lock, &ts);
|
||||||
vm->gvl.timer = 0;
|
vm->gvl.timer = 0;
|
||||||
ubf_wakeup_all_threads();
|
ubf_wakeup_all_threads();
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user