Commit 7d1a5f54 authored by Stefan Behnel's avatar Stefan Behnel

merge

parents 6d88e945 a3f75eea
...@@ -34,6 +34,8 @@ cdef class FunctionState: ...@@ -34,6 +34,8 @@ cdef class FunctionState:
cdef public dict temps_used_type cdef public dict temps_used_type
cdef public size_t temp_counter cdef public size_t temp_counter
cdef public object closure_temps
@cython.locals(n=size_t) @cython.locals(n=size_t)
cpdef new_label(self, name=*) cpdef new_label(self, name=*)
cpdef tuple get_loop_labels(self) cpdef tuple get_loop_labels(self)
......
...@@ -117,6 +117,7 @@ class FunctionState(object): ...@@ -117,6 +117,7 @@ class FunctionState(object):
self.temps_free = {} # (type, manage_ref) -> list of free vars with same type/managed status self.temps_free = {} # (type, manage_ref) -> list of free vars with same type/managed status
self.temps_used_type = {} # name -> (type, manage_ref) self.temps_used_type = {} # name -> (type, manage_ref)
self.temp_counter = 0 self.temp_counter = 0
self.closure_temps = None
# labels # labels
...@@ -270,6 +271,9 @@ class FunctionState(object): ...@@ -270,6 +271,9 @@ class FunctionState(object):
if manage_ref if manage_ref
for cname in freelist] for cname in freelist]
def init_closure_temps(self, scope):
self.closure_temps = ClosureTempAllocator(scope)
class IntConst(object): class IntConst(object):
"""Global info about a Python integer constant held by GlobalState. """Global info about a Python integer constant held by GlobalState.
...@@ -1397,7 +1401,7 @@ class PyrexCodeWriter(object): ...@@ -1397,7 +1401,7 @@ class PyrexCodeWriter(object):
class ClosureTempAllocator(object): class ClosureTempAllocator(object):
def __init__(self, klass=None): def __init__(self, klass):
self.klass = klass self.klass = klass
self.temps_allocated = {} self.temps_allocated = {}
self.temps_free = {} self.temps_free = {}
......
...@@ -4978,11 +4978,15 @@ class YieldExprNode(ExprNode): ...@@ -4978,11 +4978,15 @@ class YieldExprNode(ExprNode):
# #
# arg ExprNode the value to return from the generator # arg ExprNode the value to return from the generator
# label_name string name of the C label used for this yield # label_name string name of the C label used for this yield
# label_num integer yield label number
subexprs = ['arg'] subexprs = ['arg']
type = py_object_type type = py_object_type
label_num = 0
def analyse_types(self, env): def analyse_types(self, env):
if not self.label_num:
error(self.pos, "'yield' not supported here")
self.is_temp = 1 self.is_temp = 1
if self.arg is not None: if self.arg is not None:
self.arg.analyse_types(env) self.arg.analyse_types(env)
...@@ -5006,9 +5010,9 @@ class YieldExprNode(ExprNode): ...@@ -5006,9 +5010,9 @@ class YieldExprNode(ExprNode):
else: else:
code.put_init_to_py_none(Naming.retval_cname, py_object_type) code.put_init_to_py_none(Naming.retval_cname, py_object_type)
saved = [] saved = []
code.temp_allocator.reset() code.funcstate.closure_temps.reset()
for cname, type, manage_ref in code.funcstate.temps_in_use(): for cname, type, manage_ref in code.funcstate.temps_in_use():
save_cname = code.temp_allocator.allocate_temp(type) save_cname = code.funcstate.closure_temps.allocate_temp(type)
saved.append((cname, save_cname, type)) saved.append((cname, save_cname, type))
if type.is_pyobject: if type.is_pyobject:
code.put_xgiveref(cname) code.put_xgiveref(cname)
......
...@@ -1361,7 +1361,7 @@ class FuncDefNode(StatNode, BlockNode): ...@@ -1361,7 +1361,7 @@ class FuncDefNode(StatNode, BlockNode):
if not self.is_generator: if not self.is_generator:
self.generate_preamble(env, code) self.generate_preamble(env, code)
if self.is_generator: if self.is_generator:
code.temp_allocator = ClosureTempAllocator(lenv.scope_class.type.scope) code.funcstate.init_closure_temps(lenv.scope_class.type.scope)
resume_code = code.insertion_point() resume_code = code.insertion_point()
first_run_label = code.new_label('first_run') first_run_label = code.new_label('first_run')
code.use_label(first_run_label) code.use_label(first_run_label)
......
...@@ -1308,22 +1308,23 @@ class YieldNodeCollector(TreeVisitor): ...@@ -1308,22 +1308,23 @@ class YieldNodeCollector(TreeVisitor):
def __init__(self): def __init__(self):
super(YieldNodeCollector, self).__init__() super(YieldNodeCollector, self).__init__()
self.yields = [] self.yields = []
self.has_return = False self.returns = []
self.has_return_value = False
visit_Node = TreeVisitor.visitchildren visit_Node = TreeVisitor.visitchildren
def visit_YieldExprNode(self, node): def visit_YieldExprNode(self, node):
if self.has_return: if self.has_return_value:
error(node.pos, "'yield' outside function") error(node.pos, "'yield' outside function")
else:
self.yields.append(node) self.yields.append(node)
node.label_num = len(self.yields) node.label_num = len(self.yields)
def visit_ReturnStatNode(self, node): def visit_ReturnStatNode(self, node):
if node.value:
self.has_return_value = True
if self.yields: if self.yields:
error(collector.returns[0].pos, "'return' with argument inside generator") error(node.pos, "'return' with argument inside generator")
else: self.returns.append(node)
self.has_return = True
def visit_ClassDefNode(self, node): def visit_ClassDefNode(self, node):
pass pass
...@@ -1348,6 +1349,8 @@ class MarkClosureVisitor(CythonTransform): ...@@ -1348,6 +1349,8 @@ class MarkClosureVisitor(CythonTransform):
collector.visitchildren(node) collector.visitchildren(node)
if collector.yields: if collector.yields:
if collector.returns and not collector.has_return_value:
error(collector.returns[0].pos, "'return' inside generators not yet supported ")
node.is_generator = True node.is_generator = True
node.needs_closure = True node.needs_closure = True
node.yields = collector.yields node.yields = collector.yields
......
def foo():
yield
return 0
def bar(a):
return 0
yield
def xxx():
yield
return
yield
class Foo:
yield
_ERRORS = u"""
3:4: 'return' with argument inside generator
7:4: 'yield' outside function
11:4: 'return' inside generators not yet supported
13:0: 'yield' not supported here
16:4: 'yield' not supported here
"""
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