Commit b9575a2f authored by Stefan Behnel's avatar Stefan Behnel

use NoneCheckNode instead of explicit exception raising code in AttributeNode

parent 02e2a703
...@@ -3744,6 +3744,13 @@ class AttributeNode(ExprNode): ...@@ -3744,6 +3744,13 @@ 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
...@@ -3828,12 +3835,9 @@ class AttributeNode(ExprNode): ...@@ -3828,12 +3835,9 @@ 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, but we may need to insert # result_code contains what is needed, unless we need utility code
# a check and raise an exception if not self.obj.type.is_extension_type:
if self.obj.type.is_extension_type: if self.entry and self.entry.utility_code and self.entry.is_cmethod:
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)
...@@ -3853,11 +3857,6 @@ class AttributeNode(ExprNode): ...@@ -3853,11 +3857,6 @@ 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)
...@@ -3893,13 +3892,6 @@ class AttributeNode(ExprNode): ...@@ -3893,13 +3892,6 @@ 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("}")
#------------------------------------------------------------------- #-------------------------------------------------------------------
# #
...@@ -7483,6 +7475,8 @@ class NoneCheckNode(CoercionNode): ...@@ -7483,6 +7475,8 @@ 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
...@@ -7491,7 +7485,7 @@ class NoneCheckNode(CoercionNode): ...@@ -7491,7 +7485,7 @@ class NoneCheckNode(CoercionNode):
self.exception_message = exception_message self.exception_message = exception_message
def analyse_types(self, env): def analyse_types(self, env):
pass self.arg.analyse_types(env)
def may_be_none(self): def may_be_none(self):
return False return False
...@@ -8235,16 +8229,6 @@ static CYTHON_INLINE int __Pyx_ErrOccurredWithGIL(void) { ...@@ -8235,16 +8229,6 @@ 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