#include "Python.h"
-#include "pycore_pymath.h" // _Py_InIntegralTypeRange()
#ifdef MS_WINDOWS
# include <winsock2.h> // struct timeval
#endif
# error "unsupported time_t size"
#endif
+#if PY_TIME_T_MAX + PY_TIME_T_MIN != -1
+# error "time_t is not a two's complement integer type"
+#endif
+
+#if _PyTime_MIN + _PyTime_MAX != -1
+# error "_PyTime_t is not a two's complement integer type"
+#endif
+
static void
pytime_time_t_overflow(void)
}
assert(0.0 <= floatpart && floatpart < denominator);
- if (!_Py_InIntegralTypeRange(time_t, intpart)) {
+ /*
+ Conversion of an out-of-range value to time_t gives undefined behaviour
+ (C99 ยง6.3.1.4p1), so we must guard against it. However, checking that
+ `intpart` is in range is delicate: the obvious expression `intpart <=
+ PY_TIME_T_MAX` will first convert the value `PY_TIME_T_MAX` to a double,
+ potentially changing its value and leading to us failing to catch some
+ UB-inducing values. The code below works correctly under the mild
+ assumption that time_t is a two's complement integer type with no trap
+ representation, and that `PY_TIME_T_MIN` is within the representable
+ range of a C double.
+
+ Note: we want the `if` condition below to be true for NaNs; therefore,
+ resist any temptation to simplify by applying De Morgan's laws.
+ */
+ if (!((double)PY_TIME_T_MIN <= intpart && intpart < -(double)PY_TIME_T_MIN)) {
pytime_time_t_overflow();
return -1;
}
d = pytime_round(d, round);
(void)modf(d, &intpart);
- if (!_Py_InIntegralTypeRange(time_t, intpart)) {
+ /* See comments in pytime_double_to_denominator */
+ if (!((double)PY_TIME_T_MIN <= intpart && intpart < -(double)PY_TIME_T_MIN)) {
pytime_time_t_overflow();
return -1;
}
d *= (double)unit_to_ns;
d = pytime_round(d, round);
- if (!_Py_InIntegralTypeRange(_PyTime_t, d)) {
- pytime_overflow();
+ /* See comments in pytime_double_to_denominator */
+ if (!((double)_PyTime_MIN <= d && d < -(double)_PyTime_MIN)) {
+ pytime_time_t_overflow();
return -1;
}
_PyTime_t ns = (_PyTime_t)d;
info->monotonic = 0;
info->adjustable = 1;
if (clock_getres(CLOCK_REALTIME, &res) == 0) {
- info->resolution = res.tv_sec + res.tv_nsec * 1e-9;
+ info->resolution = (double)res.tv_sec + (double)res.tv_nsec * 1e-9;
}
else {
info->resolution = 1e-9;