Commit 350f7692 authored by Robert Bradshaw's avatar Robert Bradshaw

Fix for in/not in cascading.

parent 1175bb68
...@@ -5231,9 +5231,11 @@ class CmpNode(object): ...@@ -5231,9 +5231,11 @@ class CmpNode(object):
else: else:
negation = "" negation = ""
if op == 'in' or op == 'not_in': if op == 'in' or op == 'not_in':
assert not coerce_result code.globalstate.use_utility_code(contians_utility_code)
if self.type is PyrexTypes.py_object_type:
coerce_result = "__Pyx_PyBoolOrNull_FromLong"
if op == 'not_in': if op == 'not_in':
negation = "if (likely(%s != -1)) %s = !%s; " % ((result_code,)*3) negation = "__Pyx_NegateNonNeg"
if operand2.type is dict_type: if operand2.type is dict_type:
code.globalstate.use_utility_code( code.globalstate.use_utility_code(
raise_none_iter_error_utility_code) raise_none_iter_error_utility_code)
...@@ -5241,22 +5243,27 @@ class CmpNode(object): ...@@ -5241,22 +5243,27 @@ class CmpNode(object):
code.putln("__Pyx_RaiseNoneNotIterableError(); %s" % code.putln("__Pyx_RaiseNoneNotIterableError(); %s" %
code.error_goto(self.pos)) code.error_goto(self.pos))
code.putln("} else {") code.putln("} else {")
code.putln( method = "PyDict_Contains"
"%s = PyDict_Contains(%s, %s); %s%s" % (
result_code,
operand2.py_result(),
operand1.py_result(),
negation,
code.error_goto_if_neg(result_code, self.pos)))
code.putln("}")
else: else:
code.putln( method = "PySequence_Contains"
"%s = PySequence_Contains(%s, %s); %s%s" % ( if self.type is PyrexTypes.py_object_type:
result_code, error_clause = code.error_goto_if_null
operand2.py_result(), got_ref = "__Pyx_XGOTREF(%s); " % result_code
operand1.py_result(), else:
negation, error_clause = code.error_goto_if_neg
code.error_goto_if_neg(result_code, self.pos))) got_ref = ""
code.putln(
"%s = %s(%s(%s(%s, %s))); %s%s" % (
result_code,
coerce_result,
negation,
method,
operand2.py_result(),
operand1.py_result(),
got_ref,
error_clause(result_code, self.pos)))
if operand2.type is dict_type:
code.putln("}")
elif (operand1.type.is_pyobject elif (operand1.type.is_pyobject
and op not in ('is', 'is_not')): and op not in ('is', 'is_not')):
...@@ -5306,6 +5313,14 @@ class CmpNode(object): ...@@ -5306,6 +5313,14 @@ class CmpNode(object):
else: else:
return op return op
contians_utility_code = UtilityCode(
proto="""
static INLINE long __Pyx_NegateNonNeg(long b) { return unlikely(b < 0) ? b : !b; }
static INLINE PyObject* __Pyx_PyBoolOrNull_FromLong(long b) {
return unlikely(b < 0) ? NULL : __Pyx_PyBool_FromLong(b);
}
""")
class PrimaryCmpNode(ExprNode, CmpNode): class PrimaryCmpNode(ExprNode, CmpNode):
# Non-cascaded comparison or first comparison of # Non-cascaded comparison or first comparison of
......
...@@ -80,3 +80,18 @@ def not_in_dict(k, dict dct): ...@@ -80,3 +80,18 @@ def not_in_dict(k, dict dct):
TypeError: 'NoneType' object is not iterable TypeError: 'NoneType' object is not iterable
""" """
return k not in dct return k not in dct
def cascaded(a, b, c):
"""
>>> cascaded(1, 2, 3)
Traceback (most recent call last):
...
TypeError: argument of type 'int' is not iterable
>>> cascaded(-1, (1,2), (1,3))
True
>>> cascaded(1, (1,2), (1,3))
False
>>> cascaded(-1, (1,2), (1,0))
False
"""
return a not in b < c
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