-
-
Notifications
You must be signed in to change notification settings - Fork 31.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
bpo-30768: Recompute timeout on interrupted lock #4103
Changes from all commits
3551353
764d0cd
17aba7c
7c428b7
68e3f02
1fb576c
7232c61
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
Fix the pthread+semaphore implementation of PyThread_acquire_lock_timed() when | ||
called with timeout > 0 and intr_flag=0: recompute the timeout if | ||
sem_timedwait() is interrupted by a signal (EINTR). See also the :pep:`475`. |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -318,23 +318,66 @@ PyThread_acquire_lock_timed(PyThread_type_lock lock, PY_TIMEOUT_T microseconds, | |
sem_t *thelock = (sem_t *)lock; | ||
int status, error = 0; | ||
struct timespec ts; | ||
_PyTime_t deadline = 0; | ||
|
||
(void) error; /* silence unused-but-set-variable warning */ | ||
dprintf(("PyThread_acquire_lock_timed(%p, %lld, %d) called\n", | ||
lock, microseconds, intr_flag)); | ||
|
||
if (microseconds > 0) | ||
if (microseconds > PY_TIMEOUT_MAX) { | ||
Py_FatalError("Timeout larger than PY_TIMEOUT_MAX"); | ||
} | ||
|
||
if (microseconds > 0) { | ||
MICROSECONDS_TO_TIMESPEC(microseconds, ts); | ||
do { | ||
if (microseconds > 0) | ||
|
||
if (!intr_flag) { | ||
/* cannot overflow thanks to (microseconds > PY_TIMEOUT_MAX) | ||
check done above */ | ||
_PyTime_t timeout = _PyTime_FromNanoseconds(microseconds * 1000); | ||
deadline = _PyTime_GetMonotonicClock() + timeout; | ||
} | ||
} | ||
|
||
while (1) { | ||
if (microseconds > 0) { | ||
status = fix_status(sem_timedwait(thelock, &ts)); | ||
else if (microseconds == 0) | ||
} | ||
else if (microseconds == 0) { | ||
status = fix_status(sem_trywait(thelock)); | ||
else | ||
} | ||
else { | ||
status = fix_status(sem_wait(thelock)); | ||
} | ||
|
||
/* Retry if interrupted by a signal, unless the caller wants to be | ||
notified. */ | ||
} while (!intr_flag && status == EINTR); | ||
if (intr_flag || status != EINTR) { | ||
break; | ||
} | ||
|
||
if (microseconds > 0) { | ||
/* wait interrupted by a signal (EINTR): recompute the timeout */ | ||
_PyTime_t dt = deadline - _PyTime_GetMonotonicClock(); | ||
if (dt < 0) { | ||
status = ETIMEDOUT; | ||
break; | ||
} | ||
else if (dt > 0) { | ||
_PyTime_t realtime_deadline = _PyTime_GetSystemClock() + dt; | ||
if (_PyTime_AsTimespec(realtime_deadline, &ts) < 0) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Oh, this function raises an exception on overflow. This bug is now fixed in the new commit. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Which new commit? :-) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 4th commit of this PR, commit 7c428b7 which added the following code at the top of the function:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ok, thank you for adding the comments :-) |
||
/* Cannot occur thanks to (microseconds > PY_TIMEOUT_MAX) | ||
check done above */ | ||
Py_UNREACHABLE(); | ||
} | ||
/* no need to update microseconds value, the code only care | ||
if (microseconds > 0 or (microseconds == 0). */ | ||
} | ||
else { | ||
microseconds = 0; | ||
} | ||
} | ||
} | ||
|
||
/* Don't check the status if we're stopping because of an interrupt. */ | ||
if (!(intr_flag && status == EINTR)) { | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Which integer width does the C preprocessor use? :-) Can
0xFFFFFFFFLL * 1000
be truncated?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I didn't write this code, I only moved it.
The "LL" suffix is for "long long" which should be at least 64-bit signed integer, no?
Are you suggesting to replace the code with "0xFFFFFFFFLL < PY_LLONG_MAX / 1000" to prevent any risk of integer overflow?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah, ok. No, nevermind.