1
0
forked from M-Labs/artiq

Remove outer_final

We don't need to know whether there's a outer finally block
that's already implicit in the current break and continue
target.

Signed-off-by: Michael Birtwell <michael.birtwell@oxionics.com>
This commit is contained in:
Michael Birtwell 2022-02-22 16:49:11 +00:00 committed by Sébastien Bourdeauducq
parent 2d6215158f
commit 1b80746f48
4 changed files with 115 additions and 22 deletions

View File

@ -113,7 +113,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
self.return_target = None self.return_target = None
self.unwind_target = None self.unwind_target = None
self.catch_clauses = [] self.catch_clauses = []
self.outer_final = None
self.final_branch = None self.final_branch = None
self.function_map = dict() self.function_map = dict()
self.variable_map = dict() self.variable_map = dict()
@ -468,7 +467,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
self.append(ir.BranchIf(cond, if_true, tail), block=head) self.append(ir.BranchIf(cond, if_true, tail), block=head)
def visit_While(self, node): def visit_While(self, node):
old_outer_final, self.outer_final = self.outer_final, None
try: try:
head = self.add_block("while.head") head = self.add_block("while.head")
self.append(ir.Branch(head)) self.append(ir.Branch(head))
@ -488,7 +486,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
finally: finally:
self.break_target = old_break self.break_target = old_break
self.continue_target = old_continue self.continue_target = old_continue
self.outer_final = old_outer_final
if any(node.orelse): if any(node.orelse):
else_tail = self.add_block("while.else") else_tail = self.add_block("while.else")
@ -563,7 +560,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
assert False assert False
def visit_ForT(self, node): def visit_ForT(self, node):
old_outer_final, self.outer_final = self.outer_final, None
try: try:
iterable = self.visit(node.iter) iterable = self.visit(node.iter)
length = self.iterable_len(iterable) length = self.iterable_len(iterable)
@ -601,7 +597,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
finally: finally:
self.break_target = old_break self.break_target = old_break
self.continue_target = old_continue self.continue_target = old_continue
self.outer_final = old_outer_final
if any(node.orelse): if any(node.orelse):
else_tail = self.add_block("for.else") else_tail = self.add_block("for.else")
@ -710,9 +705,7 @@ class ARTIQIRGenerator(algorithm.Visitor):
value = return_action.append(ir.GetLocal(self.current_private_env, "$return")) value = return_action.append(ir.GetLocal(self.current_private_env, "$return"))
return_action.append(ir.Return(value)) return_action.append(ir.Return(value))
final_branch(return_action, return_proxy) final_branch(return_action, return_proxy)
else:
old_outer_final, self.outer_final = self.outer_final, final_branch
elif self.outer_final is None:
landingpad.has_cleanup = False landingpad.has_cleanup = False
# we should propagate the clauses to nested try catch blocks # we should propagate the clauses to nested try catch blocks
@ -777,7 +770,7 @@ class ARTIQIRGenerator(algorithm.Visitor):
self.continue_target = old_continue self.continue_target = old_continue
self.return_target = old_return self.return_target = old_return
if any(node.finalbody) or self.outer_final is not None: if any(node.finalbody):
# create new unwind target for cleanup # create new unwind target for cleanup
final_dispatcher = self.add_block("try.final.dispatch") final_dispatcher = self.add_block("try.final.dispatch")
final_landingpad = ir.LandingPad(cleanup) final_landingpad = ir.LandingPad(cleanup)
@ -790,16 +783,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
# if we have a while:try/finally continue must execute finally # if we have a while:try/finally continue must execute finally
# before continuing the while # before continuing the while
redirect = final_branch redirect = final_branch
elif self.outer_final is not None:
# If we have while:try/finally:try then we need to execute that finally
# before the continuing the while
# If we have try/finally:while:try then we should just continue the
# while without having to execute the finally until later
# If we have try/finally:while:try/finally:try we should execute
# one but not the other
# This is achieved by reseting self.outer_final in while and for
# loops
redirect = self.outer_final
else: else:
redirect = lambda dest, proxy: proxy.append(ir.Branch(dest)) redirect = lambda dest, proxy: proxy.append(ir.Branch(dest))
@ -860,7 +843,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
if any(node.finalbody): if any(node.finalbody):
# Finalize and continue after try statement. # Finalize and continue after try statement.
self.outer_final = old_outer_final
self.unwind_target = old_unwind self.unwind_target = old_unwind
# Exception path # Exception path
finalizer_reraise = self.add_block("finally.resume") finalizer_reraise = self.add_block("finally.resume")
@ -910,8 +892,6 @@ class ARTIQIRGenerator(algorithm.Visitor):
block.append(ir.SetLocal(final_state, "$cont", tail)) block.append(ir.SetLocal(final_state, "$cont", tail))
block.append(ir.Branch(finalizer)) block.append(ir.Branch(finalizer))
else: else:
if self.outer_final is not None:
self.unwind_target = old_unwind
self.current_block = tail = self.add_block("try.tail") self.current_block = tail = self.add_block("try.tail")
if not body.is_terminated(): if not body.is_terminated():
body.append(ir.Branch(tail)) body.append(ir.Branch(tail))

View File

@ -0,0 +1,42 @@
# RUN: %python -m artiq.compiler.testbench.jit %s >%t
# RUN: OutputCheck %s --file-to-check=%t
# REQUIRES: exceptions
def run():
loop = 0
print("start")
try:
while True:
print("loop")
try:
if loop == 0:
loop += 1
continue
func()
break
except RuntimeError:
print("except")
return False
finally:
print("finally2")
print("after-while")
finally:
print("finally1")
print("exit")
return True
def func():
print("func")
# CHECK-L: start
# CHECK-NEXT-L: loop
# CHECK-NEXT-L: finally2
# CHECK-NEXT-L: loop
# CHECK-NEXT-L: func
# CHECK-NEXT-L: finally2
# CHECK-NEXT-L: after-while
# CHECK-NEXT-L: finally1
# CHECK-NEXT-L: exit
run()

View File

@ -0,0 +1,42 @@
# RUN: %python -m artiq.compiler.testbench.jit %s >%t
# RUN: OutputCheck %s --file-to-check=%t
# REQUIRES: exceptions
def run():
print("start")
try:
try:
while True:
print("loop")
try:
print("try")
func()
print("unreachable")
return True
except RuntimeError:
print("except1")
raise
print("unreachable")
finally:
print("finally1")
print("unreachable")
return False
except RuntimeError:
print("except2")
raise
finally:
print("finally2")
return True
def func():
raise RuntimeError("Test")
# CHECK-L: start
# CHECK-NEXT-L: loop
# CHECK-NEXT-L: try
# CHECK-NEXT-L: except1
# CHECK-NEXT-L: finally1
# CHECK-NEXT-L: except2
# CHECK-NEXT-L: finally2
run()

View File

@ -0,0 +1,29 @@
# RUN: %python -m artiq.compiler.testbench.jit %s >%t
# RUN: OutputCheck %s --file-to-check=%t
# REQUIRES: exceptions
def run():
try:
while True:
try:
print("try")
func()
return True
except RuntimeError:
print("except")
return False
print("unreachable")
finally:
print("finally")
print("unreachable")
return False
def func():
pass
# CHECK-L: try
# CHECK-NOT-L: except
# CHECK-NOT-L: unreachable
# CHECK-L: finally
run()