Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
nexedi
cython
Commits
3bac3dd7
Commit
3bac3dd7
authored
Jul 18, 2017
by
Robert Bradshaw
Committed by
GitHub
Jul 18, 2017
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1777 from scoder/fix_complex_comparison
Fix complex comparison
parents
d4536e4f
99c86175
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
54 additions
and
17 deletions
+54
-17
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+8
-8
tests/run/complex_numbers_T305.pyx
tests/run/complex_numbers_T305.pyx
+46
-9
No files found.
Cython/Compiler/ExprNodes.py
View file @
3bac3dd7
...
...
@@ -1672,7 +1672,7 @@ class ImagNode(AtomicExprNode):
node
=
ImagNode
(
self
.
pos
,
value
=
self
.
value
)
if
dst_type
.
is_pyobject
:
node
.
is_temp
=
1
node
.
type
=
PyrexTypes
.
py_object
_type
node
.
type
=
Builtin
.
complex
_type
# We still need to perform normal coerce_to processing on the
# result, because we might be coercing to an extension type,
# in which case a type test node will be needed.
...
...
@@ -11866,22 +11866,22 @@ class CmpNode(object):
new_common_type
=
None
# catch general errors
if
type1
==
str_type
and
(
type2
.
is_string
or
type2
in
(
bytes_type
,
unicode_type
))
or
\
type2
==
str_type
and
(
type1
.
is_string
or
type1
in
(
bytes_type
,
unicode_type
)):
if
(
type1
==
str_type
and
(
type2
.
is_string
or
type2
in
(
bytes_type
,
unicode_type
))
or
type2
==
str_type
and
(
type1
.
is_string
or
type1
in
(
bytes_type
,
unicode_type
)
)):
error
(
self
.
pos
,
"Comparisons between bytes/unicode and str are not portable to Python 3"
)
new_common_type
=
error_type
# try to use numeric comparisons where possible
elif
type1
.
is_complex
or
type2
.
is_complex
:
if
op
not
in
(
'=='
,
'!='
)
\
and
(
type1
.
is_complex
or
type1
.
is_numeric
)
\
and
(
type2
.
is_complex
or
type2
.
is_numeric
):
if
(
op
not
in
(
'=='
,
'!='
)
and
(
type1
.
is_complex
or
type1
.
is_numeric
)
and
(
type2
.
is_complex
or
type2
.
is_numeric
)
):
error
(
self
.
pos
,
"complex types are unordered"
)
new_common_type
=
error_type
elif
type1
.
is_pyobject
:
new_common_type
=
type1
new_common_type
=
Builtin
.
complex_type
if
type1
.
subtype_of
(
Builtin
.
complex_type
)
else
py_object_type
elif
type2
.
is_pyobject
:
new_common_type
=
type2
new_common_type
=
Builtin
.
complex_type
if
type2
.
subtype_of
(
Builtin
.
complex_type
)
else
py_object_type
else
:
new_common_type
=
PyrexTypes
.
widest_numeric_type
(
type1
,
type2
)
elif
type1
.
is_numeric
and
type2
.
is_numeric
:
...
...
tests/run/complex_numbers_T305.pyx
View file @
3bac3dd7
# ticket: 305
from
cpython.object
cimport
Py_EQ
,
Py_NE
cimport
cython
cdef
class
Complex3j
:
"""
>>> Complex3j() == 3j
True
>>> Complex3j() == Complex3j()
True
>>> Complex3j() != 3j
False
>>> Complex3j() != 3
True
>>> Complex3j() != Complex3j()
False
"""
def
__richcmp__
(
a
,
b
,
int
op
):
if
op
==
Py_EQ
or
op
==
Py_NE
:
if
isinstance
(
a
,
Complex3j
):
eq
=
isinstance
(
b
,
Complex3j
)
or
b
==
3j
else
:
eq
=
isinstance
(
b
,
Complex3j
)
and
a
==
3j
return
eq
if
op
==
Py_EQ
else
not
eq
return
NotImplemented
def
test_object_conversion
(
o
):
"""
>>> test_object_conversion(2)
...
...
@@ -13,6 +39,7 @@ def test_object_conversion(o):
cdef
double
complex
b
=
o
return
(
a
,
b
)
def
test_arithmetic
(
double
complex
z
,
double
complex
w
):
"""
>>> test_arithmetic(2j, 4j)
...
...
@@ -24,6 +51,7 @@ def test_arithmetic(double complex z, double complex w):
"""
return
+
z
,
-
z
+
0
,
z
+
w
,
z
-
w
,
z
*
w
,
z
/
w
def
test_div
(
double
complex
a
,
double
complex
b
,
expected
):
"""
>>> big = 2.0**1023
...
...
@@ -34,6 +62,7 @@ def test_div(double complex a, double complex b, expected):
if
'_c99_'
not
in
__name__
:
assert
a
/
b
==
expected
,
(
a
/
b
,
expected
)
def
test_pow
(
double
complex
z
,
double
complex
w
,
tol
=
None
):
"""
Various implementations produce slightly different results...
...
...
@@ -55,6 +84,7 @@ def test_pow(double complex z, double complex w, tol=None):
else
:
return
abs
(
z
**
w
/
<
object
>
z
**
<
object
>
w
-
1
)
<
tol
def
test_int_pow
(
double
complex
z
,
int
n
,
tol
=
None
):
"""
>>> [test_int_pow(complex(0, 1), k, 1e-15) for k in range(-4, 5)]
...
...
@@ -71,6 +101,7 @@ def test_int_pow(double complex z, int n, tol=None):
else
:
return
abs
(
z
**
n
/
<
object
>
z
**
<
object
>
n
-
1
)
<
tol
@
cython
.
cdivision
(
False
)
def
test_div_by_zero
(
double
complex
z
):
"""
...
...
@@ -83,6 +114,7 @@ def test_div_by_zero(double complex z):
"""
return
1
/
z
def
test_coercion
(
int
a
,
float
b
,
double
c
,
float
complex
d
,
double
complex
e
):
"""
>>> test_coercion(1, 1.5, 2.5, 4+1j, 10j)
...
...
@@ -101,29 +133,34 @@ def test_coercion(int a, float b, double c, float complex d, double complex e):
z
=
e
;
print
z
return
z
+
a
+
b
+
c
+
d
+
e
def
test_compare
(
double
complex
a
,
double
complex
b
):
"""
>>> test_compare(3, 3)
(True, False)
(True, False
, False, False, False, True
)
>>> test_compare(3j, 3j)
(True, False)
(True, False
, True, True, True, False
)
>>> test_compare(3j, 4j)
(False, True)
(False, True
, True, False, True, True
)
>>> test_compare(3, 4)
(False, True)
(False, True
, False, False, False, True
)
"""
return
a
==
b
,
a
!=
b
return
a
==
b
,
a
!=
b
,
a
==
3j
,
3j
==
b
,
a
==
Complex3j
(),
Complex3j
()
!=
b
def
test_compare_coerce
(
double
complex
a
,
int
b
):
"""
>>> test_compare_coerce(3, 4)
(False, True)
(False, True
, False, False, False, True
)
>>> test_compare_coerce(4+1j, 4)
(False, True)
(False, True
, False, True, False, True
)
>>> test_compare_coerce(4, 4)
(True, False)
(True, False, False, False, False, True)
>>> test_compare_coerce(3j, 4)
(False, True, True, False, True, False)
"""
return
a
==
b
,
a
!=
b
return
a
==
b
,
a
!=
b
,
a
==
3j
,
4
+
1j
==
a
,
a
==
Complex3j
(),
Complex3j
()
!=
a
def
test_literal
():
"""
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment