1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
|
Origin: upstream, 473de659f78ead055de7720c191479143c1567ed
From: Haixiao Yan <haixiao.yan.cn@windriver.com>
Date: Wed, 22 Oct 2025 15:23:57 +0800
Bug-Debian: https://bugs.debian.org/1126257
Subject: fix: correct struct packing on 32-bit with _TIME_BITS=64
On 32-bit platforms with glibc time64 ABI, time_t is 64-bit wide while
`long` remains 32-bit. This causes struct timeval to use two 64-bit fields
(tv_sec, tv_usec). The previous code incorrectly packed timeout as "ll",
leading to EINVAL in setsockopt(SO_RCVTIMEO).
Use "qq" instead when m2.time_t_bits() == 64 to match the actual ABI.
Fixes: https://todo.sr.ht/~mcepl/m2crypto/374
Signed-off-by: Haixiao Yan <haixiao.yan.cn@windriver.com>
---
--- a/src/M2Crypto/SSL/timeout.py
+++ b/src/M2Crypto/SSL/timeout.py
@@ -33,10 +33,14 @@ class timeout(object):
millisec = int(self.sec * 1000 + round(float(self.microsec) / 1000))
binstr = struct.pack("l", millisec)
else:
- if m2.time_t_bits() == 32:
+ bits = m2.time_t_bits()
+ if bits == 32:
binstr = struct.pack("ii", self.sec, self.microsec)
+ elif bits == 64:
+ # handle both 64-bit and 32-bit+TIME_BITS=64
+ binstr = struct.pack("qq", self.sec, self.microsec)
else:
- binstr = struct.pack("ll", self.sec, self.microsec)
+ raise ValueError(f"Unsupported time_t_bits: {bits}")
return binstr
@@ -48,7 +52,13 @@ def struct_to_timeout(binstr: bytes) -> timeout:
sec = int(millisec / 1000)
microsec = (millisec % 1000) * 1000
else:
- (sec, microsec) = struct.unpack("ll", binstr)
+ bits = m2.time_t_bits()
+ if bits == 32:
+ (sec, microsec) = struct.unpack("ii", binstr)
+ elif bits == 64:
+ (sec, microsec) = struct.unpack("qq", binstr)
+ else:
+ raise ValueError(f"Unsupported time_t_bits: {bits}")
return timeout(sec, microsec)
@@ -56,4 +66,10 @@ def struct_size() -> int:
if sys.platform == "win32":
return struct.calcsize("l")
else:
- return struct.calcsize("ll")
+ bits = m2.time_t_bits()
+ if bits == 32:
+ return struct.calcsize("ii")
+ elif bits == 64:
+ return struct.calcsize("qq")
+ else:
+ raise ValueError(f"Unsupported time_t_bits: {bits}")
|