Commit ff08e584 authored by Stefan Behnel's avatar Stefan Behnel

reverted NoneCheckNode changes

parent 83991ecb
...@@ -3744,13 +3744,6 @@ class AttributeNode(ExprNode): ...@@ -3744,13 +3744,6 @@ class AttributeNode(ExprNode):
# methods need the normal attribute lookup # methods need the normal attribute lookup
# because they do not have struct entries # because they do not have struct entries
if entry.is_variable or entry.is_cmethod: if entry.is_variable or entry.is_cmethod:
# obj.type may be unavailable during type inference
if self.needs_none_check and self.obj.type and not env.nogil:
# FIXME: not sure if 'nogil' should be an
# error or just skip the None check
self.obj = self.obj.as_none_safe_node(
"'NoneType' object has no attribute '%s'" % self.attribute,
error = 'PyExc_AttributeError')
self.type = entry.type self.type = entry.type
self.member = entry.cname self.member = entry.cname
return return
...@@ -3835,9 +3828,12 @@ class AttributeNode(ExprNode): ...@@ -3835,9 +3828,12 @@ class AttributeNode(ExprNode):
code.error_goto_if_null(self.result(), self.pos))) code.error_goto_if_null(self.result(), self.pos)))
code.put_gotref(self.py_result()) code.put_gotref(self.py_result())
else: else:
# result_code contains what is needed, unless we need utility code # result_code contains what is needed, but we may need to insert
if not self.obj.type.is_extension_type: # a check and raise an exception
if self.entry and self.entry.utility_code and self.entry.is_cmethod: if self.obj.type.is_extension_type:
if self.needs_none_check and code.globalstate.directives['nonecheck']:
self.put_nonecheck(code)
elif self.entry and self.entry.is_cmethod and self.entry.utility_code:
# C method implemented as function call with utility code # C method implemented as function call with utility code
code.globalstate.use_utility_code(self.entry.utility_code) code.globalstate.use_utility_code(self.entry.utility_code)
...@@ -3857,6 +3853,11 @@ class AttributeNode(ExprNode): ...@@ -3857,6 +3853,11 @@ class AttributeNode(ExprNode):
self.obj.result_as(self.obj.type), self.obj.result_as(self.obj.type),
rhs.result_as(self.ctype()))) rhs.result_as(self.ctype())))
else: else:
if (self.obj.type.is_extension_type
and self.needs_none_check
and code.globalstate.directives['nonecheck']):
self.put_nonecheck(code)
select_code = self.result() select_code = self.result()
if self.type.is_pyobject and self.use_managed_ref: if self.type.is_pyobject and self.use_managed_ref:
rhs.make_owned_reference(code) rhs.make_owned_reference(code)
...@@ -3892,6 +3893,13 @@ class AttributeNode(ExprNode): ...@@ -3892,6 +3893,13 @@ class AttributeNode(ExprNode):
else: else:
code.annotate(self.pos, AnnotationItem('c_attr', 'c attribute', size=len(self.attribute))) code.annotate(self.pos, AnnotationItem('c_attr', 'c attribute', size=len(self.attribute)))
def put_nonecheck(self, code):
code.globalstate.use_utility_code(raise_noneattr_error_utility_code)
code.putln("if (%s) {" % code.unlikely("%s == Py_None") % self.obj.result_as(PyrexTypes.py_object_type))
code.putln("__Pyx_RaiseNoneAttributeError(\"%s\");" % self.attribute)
code.putln(code.error_goto(self.pos))
code.putln("}")
#------------------------------------------------------------------- #-------------------------------------------------------------------
# #
...@@ -7475,8 +7483,6 @@ class NoneCheckNode(CoercionNode): ...@@ -7475,8 +7483,6 @@ class NoneCheckNode(CoercionNode):
# raises an appropriate exception (as specified by the creating # raises an appropriate exception (as specified by the creating
# transform). # transform).
gil_message = "Exception raising on None check"
def __init__(self, arg, exception_type_cname, exception_message): def __init__(self, arg, exception_type_cname, exception_message):
CoercionNode.__init__(self, arg) CoercionNode.__init__(self, arg)
self.type = arg.type self.type = arg.type
...@@ -7485,7 +7491,7 @@ class NoneCheckNode(CoercionNode): ...@@ -7485,7 +7491,7 @@ class NoneCheckNode(CoercionNode):
self.exception_message = exception_message self.exception_message = exception_message
def analyse_types(self, env): def analyse_types(self, env):
self.arg.analyse_types(env) pass
def may_be_none(self): def may_be_none(self):
return False return False
...@@ -7500,10 +7506,6 @@ class NoneCheckNode(CoercionNode): ...@@ -7500,10 +7506,6 @@ class NoneCheckNode(CoercionNode):
return self.arg.result() return self.arg.result()
def generate_result_code(self, code): def generate_result_code(self, code):
if False and not code.globalstate.directives['nonecheck']:
# disabled - None checks are often required for safety
# and nonecheck is disabled by default - WTF!
return
code.putln( code.putln(
"if (unlikely(%s == Py_None)) {" % self.arg.result()) "if (unlikely(%s == Py_None)) {" % self.arg.result())
code.putln('PyErr_SetString(%s, "%s"); %s ' % ( code.putln('PyErr_SetString(%s, "%s"); %s ' % (
...@@ -8231,6 +8233,16 @@ static CYTHON_INLINE int __Pyx_ErrOccurredWithGIL(void) { ...@@ -8231,6 +8233,16 @@ static CYTHON_INLINE int __Pyx_ErrOccurredWithGIL(void) {
#------------------------------------------------------------------------------------ #------------------------------------------------------------------------------------
raise_noneattr_error_utility_code = UtilityCode(
proto = """
static CYTHON_INLINE void __Pyx_RaiseNoneAttributeError(const char* attrname);
""",
impl = '''
static CYTHON_INLINE void __Pyx_RaiseNoneAttributeError(const char* attrname) {
PyErr_Format(PyExc_AttributeError, "'NoneType' object has no attribute '%s'", attrname);
}
''')
raise_noneindex_error_utility_code = UtilityCode( raise_noneindex_error_utility_code = UtilityCode(
proto = """ proto = """
static CYTHON_INLINE void __Pyx_RaiseNoneIndexingError(void); static CYTHON_INLINE void __Pyx_RaiseNoneIndexingError(void);
......
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