summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAntoine Pitrou <pitrou@free.fr>2017-06-30 10:54:54 +0200
committerGitHub <noreply@github.com>2017-06-30 10:54:54 +0200
commita45a99b47ff241ce0ae2f0bba59b89e4e012d47c (patch)
tree3a1305b56aa6af60ccc2bfc698602a29a57b73e1
parent12536bd261ba95cd2748f3d7d47768742a6ffa7a (diff)
downloadcpython-git-a45a99b47ff241ce0ae2f0bba59b89e4e012d47c.tar.gz
[2.7] bpo-30807: signal.setitimer() may disable the timer by mistake (GH-2493) (#2499)
* bpo-30807: signal.setitimer() may disable the timer by mistake * Add NEWS blurb. (cherry picked from commit 729780a810bbcb12b245a1b652302a601fc9f6fd)
-rw-r--r--Lib/test/test_signal.py10
-rw-r--r--Misc/NEWS.d/next/Library/2017-06-29-22-04-44.bpo-30807.sLtjY-.rst6
-rw-r--r--Modules/signalmodule.c4
3 files changed, 20 insertions, 0 deletions
diff --git a/Lib/test/test_signal.py b/Lib/test/test_signal.py
index 7483f64123..27054f17ba 100644
--- a/Lib/test/test_signal.py
+++ b/Lib/test/test_signal.py
@@ -490,6 +490,16 @@ class ItimerTest(unittest.TestCase):
# and the handler should have been called
self.assertEqual(self.hndl_called, True)
+ def test_setitimer_tiny(self):
+ # bpo-30807: C setitimer() takes a microsecond-resolution interval.
+ # Check that float -> timeval conversion doesn't round
+ # the interval down to zero, which would disable the timer.
+ self.itimer = signal.ITIMER_REAL
+ signal.setitimer(self.itimer, 1e-6)
+ time.sleep(1)
+ self.assertEqual(self.hndl_called, True)
+
+
def test_main():
test_support.run_unittest(BasicSignalTests, InterProcessSignalTests,
WakeupFDTests, WakeupSignalTests,
diff --git a/Misc/NEWS.d/next/Library/2017-06-29-22-04-44.bpo-30807.sLtjY-.rst b/Misc/NEWS.d/next/Library/2017-06-29-22-04-44.bpo-30807.sLtjY-.rst
new file mode 100644
index 0000000000..ce6f48a61f
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2017-06-29-22-04-44.bpo-30807.sLtjY-.rst
@@ -0,0 +1,6 @@
+signal.setitimer() may disable the timer when passed a tiny value.
+
+Tiny values (such as 1e-6) are valid non-zero values for setitimer(), which
+is specified as taking microsecond-resolution intervals. However, on some
+platform, our conversion routine could convert 1e-6 into a zero interval,
+therefore disabling the timer instead of (re-)scheduling it.
diff --git a/Modules/signalmodule.c b/Modules/signalmodule.c
index 1d7ba4ba48..c7bf1f0853 100644
--- a/Modules/signalmodule.c
+++ b/Modules/signalmodule.c
@@ -111,6 +111,10 @@ timeval_from_double(double d, struct timeval *tv)
{
tv->tv_sec = floor(d);
tv->tv_usec = fmod(d, 1.0) * 1000000.0;
+ /* Don't disable the timer if the computation above rounds down to zero. */
+ if (d > 0.0 && tv->tv_sec == 0 && tv->tv_usec == 0) {
+ tv->tv_usec = 1;
+ }
}
Py_LOCAL_INLINE(double)