Commit 998f767f authored by Mark Florisson's avatar Mark Florisson

Allow nogil None checking for memoryview slices

parent 9076b8a4
...@@ -8643,6 +8643,7 @@ class PrimaryCmpNode(ExprNode, CmpNode): ...@@ -8643,6 +8643,7 @@ class PrimaryCmpNode(ExprNode, CmpNode):
child_attrs = ['operand1', 'operand2', 'cascade'] child_attrs = ['operand1', 'operand2', 'cascade']
cascade = None cascade = None
is_memslice_nonecheck = False
def infer_type(self, env): def infer_type(self, env):
# TODO: Actually implement this (after merging with -unstable). # TODO: Actually implement this (after merging with -unstable).
...@@ -8666,6 +8667,10 @@ class PrimaryCmpNode(ExprNode, CmpNode): ...@@ -8666,6 +8667,10 @@ class PrimaryCmpNode(ExprNode, CmpNode):
if self.cascade: if self.cascade:
error(self.pos, "Cascading comparison not yet supported for cpp types.") error(self.pos, "Cascading comparison not yet supported for cpp types.")
return return
if self.analyse_memoryviewslice_comparison(env):
return
if self.cascade: if self.cascade:
self.cascade.analyse_types(env) self.cascade.analyse_types(env)
...@@ -8744,6 +8749,19 @@ class PrimaryCmpNode(ExprNode, CmpNode): ...@@ -8744,6 +8749,19 @@ class PrimaryCmpNode(ExprNode, CmpNode):
self.operand2 = self.operand2.coerce_to(func_type.args[1].type, env) self.operand2 = self.operand2.coerce_to(func_type.args[1].type, env)
self.type = func_type.return_type self.type = func_type.return_type
def analyse_memoryviewslice_comparison(self, env):
have_none = self.operand1.is_none or self.operand2.is_none
have_slice = (self.operand1.type.is_memoryviewslice or
self.operand2.type.is_memoryviewslice)
ops = ('==', '!=', 'is', 'is_not')
if have_slice and have_none and self.operator in ops:
self.is_pycmp = False
self.type = PyrexTypes.c_bint_type
self.is_memslice_nonecheck = True
return True
return False
def has_python_operands(self): def has_python_operands(self):
return (self.operand1.type.is_pyobject return (self.operand1.type.is_pyobject
or self.operand2.type.is_pyobject) or self.operand2.type.is_pyobject)
...@@ -8781,10 +8799,18 @@ class PrimaryCmpNode(ExprNode, CmpNode): ...@@ -8781,10 +8799,18 @@ class PrimaryCmpNode(ExprNode, CmpNode):
self.operand2.result(), self.operand2.result(),
self.operand1.result()) self.operand1.result())
else: else:
result1 = self.operand1.result()
result2 = self.operand2.result()
if self.is_memslice_nonecheck:
if self.operand1.type.is_memoryviewslice:
result1 = "((PyObject *) %s.memview)" % result1
else:
result2 = "((PyObject *) %s.memview)" % result2
return "(%s %s %s)" % ( return "(%s %s %s)" % (
self.operand1.result(), result1,
self.c_operator(self.operator), self.c_operator(self.operator),
self.operand2.result()) result2)
def generate_evaluation_code(self, code): def generate_evaluation_code(self, code):
self.operand1.generate_evaluation_code(code) self.operand1.generate_evaluation_code(code)
......
...@@ -2180,6 +2180,21 @@ def test_noneslice_del(): ...@@ -2180,6 +2180,21 @@ def test_noneslice_del():
del m del m
print m print m
@testcase
def test_noneslice_nogil_check_none(double[:] m):
"""
>>> test_noneslice_nogil_check_none(None)
(True, False)
"""
cdef bint is_none = False
cdef bint not_none = True
with nogil:
is_none = m is None and None is m and m == None and None == m
not_none = m is not None and None is not m and m != None and None != m
return is_none, not_none
def get_int(): def get_int():
return 10 return 10
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment