Commit f21c0e97 authored by Stefan Behnel's avatar Stefan Behnel

discard lots of useless code and overhead from try-finally statements that...

discard lots of useless code and overhead from try-finally statements that don't need to handle Python exceptions (including C-only with-blocks)
parent 1ae99fe1
...@@ -6068,10 +6068,7 @@ class TryExceptStatNode(StatNode): ...@@ -6068,10 +6068,7 @@ class TryExceptStatNode(StatNode):
exc_save_vars = [code.funcstate.allocate_temp(py_object_type, False) exc_save_vars = [code.funcstate.allocate_temp(py_object_type, False)
for _ in xrange(3)] for _ in xrange(3)]
code.putln("{") code.putln("{")
code.putln("__Pyx_ExceptionSave(%s);" % save_exc = code.insertion_point()
', '.join(['&%s' % var for var in exc_save_vars]))
for var in exc_save_vars:
code.put_xgotref(var)
code.putln( code.putln(
"/*try:*/ {") "/*try:*/ {")
code.return_label = try_return_label code.return_label = try_return_label
...@@ -6081,6 +6078,28 @@ class TryExceptStatNode(StatNode): ...@@ -6081,6 +6078,28 @@ class TryExceptStatNode(StatNode):
code.putln( code.putln(
"}") "}")
temps_to_clean_up = code.funcstate.all_free_managed_temps() temps_to_clean_up = code.funcstate.all_free_managed_temps()
can_raise = code.label_used(our_error_label)
if can_raise:
# inject code before the try block to save away the exception state
save_exc.putln("__Pyx_ExceptionSave(%s);" %
', '.join(['&%s' % var for var in exc_save_vars]))
for var in exc_save_vars:
save_exc.put_xgotref(var)
def restore_saved_exception():
for name in exc_save_vars:
code.put_xgiveref(name)
code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(exc_save_vars))
else:
# try block cannot raise exceptions, but we had to allocate the temps above,
# so just keep the C compiler from complaining about them being unused
save_exc.putln("if (%s); else {/*mark used*/};" % '||'.join(exc_save_vars))
def restore_saved_exception():
pass
code.error_label = except_error_label code.error_label = except_error_label
code.return_label = except_return_label code.return_label = except_return_label
if self.else_clause: if self.else_clause:
...@@ -6089,50 +6108,37 @@ class TryExceptStatNode(StatNode): ...@@ -6089,50 +6108,37 @@ class TryExceptStatNode(StatNode):
self.else_clause.generate_execution_code(code) self.else_clause.generate_execution_code(code)
code.putln( code.putln(
"}") "}")
if can_raise:
for var in exc_save_vars: for var in exc_save_vars:
code.put_xdecref_clear(var, py_object_type) code.put_xdecref_clear(var, py_object_type)
code.put_goto(try_end_label) code.put_goto(try_end_label)
if code.label_used(try_return_label):
code.put_label(try_return_label)
for var in exc_save_vars:
code.put_xgiveref(var)
code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(exc_save_vars))
code.put_goto(old_return_label)
code.put_label(our_error_label) code.put_label(our_error_label)
for temp_name, type in temps_to_clean_up: for temp_name, temp_type in temps_to_clean_up:
code.put_xdecref_clear(temp_name, type) code.put_xdecref_clear(temp_name, temp_type)
for except_clause in self.except_clauses: for except_clause in self.except_clauses:
except_clause.generate_handling_code(code, except_end_label) except_clause.generate_handling_code(code, except_end_label)
if not self.has_default_clause:
error_label_used = code.label_used(except_error_label) code.put_goto(except_error_label)
if error_label_used or not self.has_default_clause:
if error_label_used: for exit_label, old_label in [(except_error_label, old_error_label),
code.put_label(except_error_label) (try_break_label, old_break_label),
for var in exc_save_vars: (try_continue_label, old_continue_label),
code.put_xgiveref(var) (try_return_label, old_return_label),
code.putln("__Pyx_ExceptionReset(%s);" % (except_return_label, old_return_label)]:
', '.join(exc_save_vars))
code.put_goto(old_error_label)
for exit_label, old_label in zip(
[try_break_label, try_continue_label, except_return_label],
[old_break_label, old_continue_label, old_return_label]):
if code.label_used(exit_label): if code.label_used(exit_label):
if not code.label_used(try_end_label):
code.put_goto(try_end_label)
code.put_label(exit_label) code.put_label(exit_label)
for var in exc_save_vars: restore_saved_exception()
code.put_xgiveref(var)
code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(exc_save_vars))
code.put_goto(old_label) code.put_goto(old_label)
if code.label_used(except_end_label): if code.label_used(except_end_label):
if not code.label_used(try_end_label):
code.put_goto(try_end_label)
code.put_label(except_end_label) code.put_label(except_end_label)
for var in exc_save_vars: restore_saved_exception()
code.put_xgiveref(var) if code.label_used(try_end_label):
code.putln("__Pyx_ExceptionReset(%s);" %
', '.join(exc_save_vars))
code.put_label(try_end_label) code.put_label(try_end_label)
code.putln("}") code.putln("}")
......
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