Commit e6d53b23 authored by Stefan Behnel's avatar Stefan Behnel

fix ticket 790: reference leak during dict iteration

parent 284a2423
...@@ -5314,24 +5314,21 @@ class DictIterationNextNode(Node): ...@@ -5314,24 +5314,21 @@ class DictIterationNextNode(Node):
type = PyrexTypes.c_bint_type) type = PyrexTypes.c_bint_type)
def analyse_expressions(self, env): def analyse_expressions(self, env):
import UtilNodes import ExprNodes
self.dict_obj.analyse_types(env) self.dict_obj.analyse_types(env)
self.expected_size.analyse_types(env) self.expected_size.analyse_types(env)
if self.pos_index_var: self.pos_index_var.analyse_types(env) if self.pos_index_var: self.pos_index_var.analyse_types(env)
if self.key_target: if self.key_target:
self.key_target.analyse_target_types(env) self.key_target.analyse_target_types(env)
self.key_ref = UtilNodes.ResultRefNode(pos=self.key_target.pos, is_temp=True, self.key_ref = ExprNodes.TempNode(self.key_target.pos, PyrexTypes.py_object_type)
type=PyrexTypes.py_object_type)
self.coerced_key_var = self.key_ref.coerce_to(self.key_target.type, env) self.coerced_key_var = self.key_ref.coerce_to(self.key_target.type, env)
if self.value_target: if self.value_target:
self.value_target.analyse_target_types(env) self.value_target.analyse_target_types(env)
self.value_ref = UtilNodes.ResultRefNode(pos=self.value_target.pos, is_temp=True, self.value_ref = ExprNodes.TempNode(self.value_target.pos, type=PyrexTypes.py_object_type)
type=PyrexTypes.py_object_type)
self.coerced_value_var = self.value_ref.coerce_to(self.value_target.type, env) self.coerced_value_var = self.value_ref.coerce_to(self.value_target.type, env)
if self.tuple_target: if self.tuple_target:
self.tuple_target.analyse_target_types(env) self.tuple_target.analyse_target_types(env)
self.tuple_ref = UtilNodes.ResultRefNode(pos=self.tuple_target.pos, is_temp=True, self.tuple_ref = ExprNodes.TempNode(self.tuple_target.pos, PyrexTypes.py_object_type)
type=PyrexTypes.py_object_type)
self.coerced_tuple_var = self.tuple_ref.coerce_to(self.tuple_target.type, env) self.coerced_tuple_var = self.tuple_ref.coerce_to(self.tuple_target.type, env)
self.is_dict_flag.analyse_types(env) self.is_dict_flag.analyse_types(env)
...@@ -5350,10 +5347,9 @@ class DictIterationNextNode(Node): ...@@ -5350,10 +5347,9 @@ class DictIterationNextNode(Node):
if target is None: if target is None:
addr = 'NULL' addr = 'NULL'
else: else:
temp = code.funcstate.allocate_temp(PyrexTypes.py_object_type, True) assignments.append((var, result, target))
var.result_code = temp var.allocate(code)
assignments.append((temp, result, target)) addr = '&%s' % var.result()
addr = '&%s' % temp
temp_addresses.append(addr) temp_addresses.append(addr)
result_temp = code.funcstate.allocate_temp(PyrexTypes.c_int_type, False) result_temp = code.funcstate.allocate_temp(PyrexTypes.c_int_type, False)
...@@ -5372,17 +5368,13 @@ class DictIterationNextNode(Node): ...@@ -5372,17 +5368,13 @@ class DictIterationNextNode(Node):
code.funcstate.release_temp(result_temp) code.funcstate.release_temp(result_temp)
# evaluate all coercions before the assignments # evaluate all coercions before the assignments
for temp, result, target in assignments: for var, result, target in assignments:
code.put_gotref(temp) code.put_gotref(var.result())
result.generate_evaluation_code(code) result.generate_evaluation_code(code)
if not result.type.is_pyobject:
code.put_decref_clear(temp, PyrexTypes.py_object_type)
code.funcstate.release_temp(temp)
for temp, result, target in assignments: for var, result, target in assignments:
target.generate_assignment_code(result, code) target.generate_assignment_code(result, code)
if result.type.is_pyobject: var.release(code)
code.funcstate.release_temp(temp)
def ForStatNode(pos, **kw): def ForStatNode(pos, **kw):
if 'iterator' in kw: if 'iterator' in kw:
......
# mode: run
# tag: dictiter
def iteritems_unpack(dict the_dict):
"""
>>> d = {(1,2): (3,4), (5,6): (7,8)}
>>> iteritems_unpack(d)
[(1, 2, 3, 4), (5, 6, 7, 8)]
"""
return sorted([ (a,b,c,d) for (a,b), (c,d) in the_dict.iteritems() ])
def itervalues_unpack(dict the_dict):
"""
>>> d = {1: (3,4), 2: (7,8)}
>>> itervalues_unpack(d)
[(3, 4), (7, 8)]
"""
return [(a,b) for a,b in the_dict.itervalues() ]
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