Commit 91d1ed40 authored by Stefan Behnel's avatar Stefan Behnel

clean up extended iterable unpacking code and avoid CPython specific code in non-CPython

parent a69a2c71
...@@ -5022,66 +5022,93 @@ class SequenceNode(ExprNode): ...@@ -5022,66 +5022,93 @@ class SequenceNode(ExprNode):
for i, arg in enumerate(self.args): for i, arg in enumerate(self.args):
if arg.is_starred: if arg.is_starred:
starred_target = self.unpacked_items[i] starred_target = self.unpacked_items[i]
fixed_args_left = self.args[:i] unpacked_fixed_items_left = self.unpacked_items[:i]
fixed_args_right = self.args[i+1:] unpacked_fixed_items_right = self.unpacked_items[i+1:]
break break
else:
assert False
iterator_temp = code.funcstate.allocate_temp(py_object_type, manage_ref=True) iterator_temp = None
code.putln( if unpacked_fixed_items_left:
"%s = PyObject_GetIter(%s); %s" % ( iterator_temp = code.funcstate.allocate_temp(py_object_type, manage_ref=True)
code.putln("%s = PyObject_GetIter(%s); %s" % (
iterator_temp, iterator_temp,
rhs.py_result(), rhs.py_result(),
code.error_goto_if_null(iterator_temp, self.pos))) code.error_goto_if_null(iterator_temp, self.pos)))
code.put_gotref(iterator_temp) code.put_gotref(iterator_temp)
rhs.generate_disposal_code(code) rhs.generate_disposal_code(code)
for item in self.unpacked_items: code.globalstate.use_utility_code(
item.allocate(code) UtilityCode.load_cached("UnpackItem", "ObjectHandling.c"))
code.globalstate.use_utility_code(unpacking_utility_code) for i, item in enumerate(unpacked_fixed_items_left):
for i in range(len(fixed_args_left)): item.allocate(code)
item = self.unpacked_items[i] code.putln("%s = __Pyx_UnpackItem(%s, %d); %s" % (
unpack_code = "__Pyx_UnpackItem(%s, %d)" % ( item.result(), iterator_temp, i,
iterator_temp, i)
code.putln(
"%s = %s; %s" % (
item.result(),
typecast(item.ctype(), py_object_type, unpack_code),
code.error_goto_if_null(item.result(), self.pos))) code.error_goto_if_null(item.result(), self.pos)))
code.put_gotref(item.py_result()) code.put_gotref(item.py_result())
value_node = self.coerced_unpacked_items[i] value_node = self.coerced_unpacked_items[i]
value_node.generate_evaluation_code(code) value_node.generate_evaluation_code(code)
starred_target.allocate(code)
target_list = starred_target.result() target_list = starred_target.result()
code.putln("%s = PySequence_List(%s); %s" % ( code.putln("%s = PySequence_List(%s); %s" % (
target_list, iterator_temp, target_list,
iterator_temp or rhs.py_result(),
code.error_goto_if_null(target_list, self.pos))) code.error_goto_if_null(target_list, self.pos)))
code.put_gotref(target_list) code.put_gotref(target_list)
if fixed_args_right:
if iterator_temp:
code.put_decref_clear(iterator_temp, py_object_type)
code.funcstate.release_temp(iterator_temp)
else:
rhs.generate_disposal_code(code)
if unpacked_fixed_items_right:
code.globalstate.use_utility_code(raise_need_more_values_to_unpack) code.globalstate.use_utility_code(raise_need_more_values_to_unpack)
unpacked_right_args = self.unpacked_items[-len(fixed_args_right):] length_temp = code.funcstate.allocate_temp(PyrexTypes.c_py_ssize_t_type, manage_ref=False)
code.putln("if (unlikely(PyList_GET_SIZE(%s) < %d)) {" % ( code.putln('%s = PyList_GET_SIZE(%s);' % (length_temp, target_list))
(target_list, len(unpacked_right_args)))) code.putln("if (unlikely(%s < %d)) {" % (length_temp, len(unpacked_fixed_items_right)))
code.put("__Pyx_RaiseNeedMoreValuesError(%d+PyList_GET_SIZE(%s)); %s" % ( code.putln("__Pyx_RaiseNeedMoreValuesError(%d+%s); %s" % (
len(fixed_args_left), target_list, len(unpacked_fixed_items_left), length_temp,
code.error_goto(self.pos))) code.error_goto(self.pos)))
code.putln('}') code.putln('}')
for i, (arg, coerced_arg) in enumerate(zip(unpacked_right_args[::-1],
self.coerced_unpacked_items[::-1])): for item in unpacked_fixed_items_right[::-1]:
item.allocate(code)
code.putln('#if CYTHON_COMPILING_IN_CPYTHON')
for i, (item, coerced_arg) in enumerate(zip(unpacked_fixed_items_right[::-1],
self.coerced_unpacked_items[::-1])):
code.putln( code.putln(
"%s = PyList_GET_ITEM(%s, PyList_GET_SIZE(%s)-1); " % ( "%s = PyList_GET_ITEM(%s, %s-%d); " % (
arg.py_result(), item.py_result(), target_list, length_temp, i+1))
target_list, target_list))
# resize the list the hard way # resize the list the hard way
code.putln("((PyVarObject*)%s)->ob_size--;" % target_list) code.putln("((PyVarObject*)%s)->ob_size--;" % target_list)
code.put_gotref(arg.py_result()) code.put_gotref(item.py_result())
coerced_arg.generate_evaluation_code(code) coerced_arg.generate_evaluation_code(code)
code.put_decref_clear(iterator_temp, py_object_type) code.putln('#else')
code.funcstate.release_temp(iterator_temp)
for i in range(len(self.args)): for i, (item, coerced_arg) in enumerate(zip(unpacked_fixed_items_right[::-1],
self.args[i].generate_assignment_code( self.coerced_unpacked_items[::-1])):
self.coerced_unpacked_items[i], code) code.putln(
"%s = PySequence_GetItem(%s, %s-%d); " % (
item.py_result(), target_list, length_temp, i+1))
# resize the list the hard way
code.put_gotref(item.py_result())
coerced_arg.generate_evaluation_code(code)
sublist_temp = code.funcstate.allocate_temp(py_object_type, manage_ref=True)
code.putln('%s = PyList_GetSlice(%s, 0, %s-%d); %s' % (
sublist_temp, target_list, length_temp, len(unpacked_fixed_items_right),
code.error_goto_if_null(sublist_temp, self.pos)))
code.put_gotref(sublist_temp)
code.funcstate.release_temp(length_temp)
code.put_decref(target_list, py_object_type)
code.putln('%s = %s; %s = NULL;' % (target_list, sublist_temp, sublist_temp))
code.funcstate.release_temp(sublist_temp)
code.putln('#endif')
for i, arg in enumerate(self.args):
arg.generate_assignment_code(self.coerced_unpacked_items[i], code)
def annotate(self, code): def annotate(self, code):
for arg in self.args: for arg in self.args:
...@@ -10099,7 +10126,6 @@ raise_need_more_values_to_unpack = UtilityCode.load_cached("RaiseNeedMoreValuesT ...@@ -10099,7 +10126,6 @@ raise_need_more_values_to_unpack = UtilityCode.load_cached("RaiseNeedMoreValuesT
#------------------------------------------------------------------------------------ #------------------------------------------------------------------------------------
tuple_unpacking_error_code = UtilityCode.load_cached("UnpackTupleError", "ObjectHandling.c") tuple_unpacking_error_code = UtilityCode.load_cached("UnpackTupleError", "ObjectHandling.c")
unpacking_utility_code = UtilityCode.load_cached("UnpackItem", "ObjectHandling.c")
iternext_unpacking_end_utility_code = UtilityCode.load_cached("UnpackItemEndCheck", "ObjectHandling.c") iternext_unpacking_end_utility_code = UtilityCode.load_cached("UnpackItemEndCheck", "ObjectHandling.c")
#------------------------------------------------------------------------------------ #------------------------------------------------------------------------------------
......
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