Skip to content

Commit 2f2c377

Browse files
authored
[mypyc] Fix division of negative tagged int (#11168)
The rounding was incorrect for some negative values, because of a bug related to the tagged integer logic. For example, -5 // 2 evaluated to -4 instead of -3 (ouch). Change the rounding implementation to not use shifted integers. Also add more exhaustive testing and fix a broken integer test case.
1 parent 58fb493 commit 2f2c377

File tree

3 files changed

+32
-15
lines changed

3 files changed

+32
-15
lines changed

mypyc/lib-rt/int_ops.c

+6-5
Original file line numberDiff line numberDiff line change
@@ -190,16 +190,17 @@ CPyTagged CPyTagged_Multiply(CPyTagged left, CPyTagged right) {
190190
}
191191

192192
CPyTagged CPyTagged_FloorDivide(CPyTagged left, CPyTagged right) {
193-
if (CPyTagged_CheckShort(left) && CPyTagged_CheckShort(right)
193+
if (CPyTagged_CheckShort(left)
194+
&& CPyTagged_CheckShort(right)
194195
&& !CPyTagged_MaybeFloorDivideFault(left, right)) {
195-
Py_ssize_t result = ((Py_ssize_t)left / CPyTagged_ShortAsSsize_t(right)) & ~1;
196+
Py_ssize_t result = CPyTagged_ShortAsSsize_t(left) / CPyTagged_ShortAsSsize_t(right);
196197
if (((Py_ssize_t)left < 0) != (((Py_ssize_t)right) < 0)) {
197-
if (result / 2 * right != left) {
198+
if (result * right != left) {
198199
// Round down
199-
result -= 2;
200+
result--;
200201
}
201202
}
202-
return result;
203+
return result << 1;
203204
}
204205
PyObject *left_obj = CPyTagged_AsObject(left);
205206
PyObject *right_obj = CPyTagged_AsObject(right);

mypyc/test-data/fixtures/ir.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -286,7 +286,8 @@ def next(i: Iterator[T]) -> T: pass
286286
def next(i: Iterator[T], default: T) -> T: pass
287287
def hash(o: object) -> int: ...
288288
def globals() -> Dict[str, Any]: ...
289-
def setattr(object: Any, name: str, value: Any) -> None: ...
289+
def getattr(obj: object, name: str) -> Any: ...
290+
def setattr(obj: object, name: str, value: Any) -> None: ...
290291
def enumerate(x: Iterable[T]) -> Iterator[Tuple[int, T]]: ...
291292
@overload
292293
def zip(x: Iterable[T], y: Iterable[S]) -> Iterator[Tuple[T, S]]: ...

mypyc/test-data/run-integers.test

+24-9
Original file line numberDiff line numberDiff line change
@@ -131,6 +131,14 @@ assert neg(-9223372036854775807) == 9223372036854775807
131131
assert neg(9223372036854775808) == -9223372036854775808
132132
assert neg(-9223372036854775808) == 9223372036854775808
133133

134+
[case testIsinstanceIntAndNotBool]
135+
def test_isinstance_int_and_not_bool(value: object) -> bool:
136+
return isinstance(value, int) and not isinstance(value, bool)
137+
[file driver.py]
138+
from native import test_isinstance_int_and_not_bool
139+
assert test_isinstance_int_and_not_bool(True) == False
140+
assert test_isinstance_int_and_not_bool(1) == True
141+
134142
[case testIntOps]
135143
def check_and(x: int, y: int) -> None:
136144
# eval() can be trusted to calculate expected result
@@ -157,15 +165,6 @@ def check_bitwise(x: int, y: int) -> None:
157165
check_or(ll, rr)
158166
check_xor(ll, rr)
159167

160-
[case testIsinstanceIntAndNotBool]
161-
def test_isinstance_int_and_not_bool(value: object) -> bool:
162-
return isinstance(value, int) and not isinstance(value, bool)
163-
[file driver.py]
164-
from native import test_isinstance_int_and_not_bool
165-
assert test_isinstance_int_and_not_bool(True) == False
166-
assert test_isinstance_int_and_not_bool(1) == True
167-
168-
169168
SHIFT = 30
170169
DIGIT0a = 615729753
171170
DIGIT0b = 832796681
@@ -198,6 +197,10 @@ def test_and_or_xor() -> None:
198197
check_bitwise(BIG_SHORT, DIGIT0a + DIGIT1a + DIGIT2a)
199198
check_bitwise(BIG_SHORT, DIGIT0a + DIGIT1a + DIGIT2a + DIGIT50)
200199

200+
for x in range(-25, 25):
201+
for y in range(-25, 25):
202+
check_bitwise(x, y)
203+
201204
def test_bitwise_inplace() -> None:
202205
# Basic sanity checks; these should use the same code as the non-in-place variants
203206
for x, y in (DIGIT0a, DIGIT1a), (DIGIT2a, DIGIT0a + DIGIT2b):
@@ -328,3 +331,15 @@ def test_int_as_bool() -> None:
328331
for x in 1, 55, -1, -7, 1 << 50, 1 << 101, -(1 << 50), -(1 << 101):
329332
assert is_true(x)
330333
assert not is_false(x)
334+
335+
def test_divide() -> None:
336+
for x in range(-100, 100):
337+
for y in range(-100, 100):
338+
if y != 0:
339+
assert x // y == getattr(x, "__floordiv__")(y)
340+
341+
def test_mod() -> None:
342+
for x in range(-100, 100):
343+
for y in range(-100, 100):
344+
if y != 0:
345+
assert x % y == getattr(x, "__mod__")(y)

0 commit comments

Comments
 (0)