Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
nexedi
cython
Commits
f98829fb
Commit
f98829fb
authored
Mar 23, 2012
by
scoder
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #98 from scoder/_unpacking_loop
optimised tuple/list/iterable unpacking
parents
8e396c4b
cb07d807
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
229 additions
and
29 deletions
+229
-29
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+66
-29
tests/run/unpack.pyx
tests/run/unpack.pyx
+51
-0
tests/run/unpack_fused.pyx
tests/run/unpack_fused.pyx
+112
-0
No files found.
Cython/Compiler/ExprNodes.py
View file @
f98829fb
...
...
@@ -4877,6 +4877,7 @@ class SequenceNode(ExprNode):
special_unpack
=
(
rhs
.
type
is
py_object_type
or
rhs
.
type
in
(
tuple_type
,
list_type
)
or
not
rhs
.
type
.
is_builtin_type
)
long_enough_for_a_loop
=
len
(
self
.
unpacked_items
)
>
3
if
special_unpack
:
tuple_check
=
'likely(PyTuple_CheckExact(%s))'
%
rhs
.
py_result
()
list_check
=
'PyList_CheckExact(%s)'
%
rhs
.
py_result
()
...
...
@@ -4889,29 +4890,32 @@ class SequenceNode(ExprNode):
else
:
sequence_types
=
[
'Tuple'
,
'List'
]
sequence_type_test
=
"(%s) || (%s)"
%
(
tuple_check
,
list_check
)
code
.
putln
(
"#if CYTHON_COMPILING_IN_CPYTHON"
)
code
.
putln
(
"if (%s) {"
%
sequence_type_test
)
code
.
putln
(
"PyObject* sequence = %s;"
%
rhs
.
py_result
())
if
len
(
sequence_types
)
==
2
:
code
.
putln
(
"if (likely(Py%s_CheckExact(sequence))) {"
%
sequence_types
[
0
])
self
.
generate_special_parallel_unpacking_code
(
code
,
sequence_types
[
0
])
self
.
generate_special_parallel_unpacking_code
(
code
,
sequence_types
[
0
],
use_loop
=
long_enough_for_a_loop
and
sequence_types
[
0
]
!=
'Tuple'
)
if
len
(
sequence_types
)
==
2
:
code
.
putln
(
"} else {"
)
self
.
generate_special_parallel_unpacking_code
(
code
,
sequence_types
[
1
])
self
.
generate_special_parallel_unpacking_code
(
code
,
sequence_types
[
1
],
use_loop
=
long_enough_for_a_loop
)
code
.
putln
(
"}"
)
for
item
in
self
.
unpacked_items
:
code
.
put_incref
(
item
.
result
(),
item
.
ctype
())
rhs
.
generate_disposal_code
(
code
)
code
.
putln
(
"} else {"
)
else
:
code
.
putln
(
"{"
)
code
.
putln
(
"} else"
)
code
.
putln
(
"#endif"
)
code
.
putln
(
"{"
)
if
special_unpack
and
rhs
.
type
is
tuple_type
:
code
.
globalstate
.
use_utility_code
(
tuple_unpacking_error_code
)
code
.
putln
(
"__Pyx_UnpackTupleError(%s, %s);"
%
(
rhs
.
py_result
(),
len
(
self
.
args
)))
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
else
:
self
.
generate_generic_parallel_unpacking_code
(
code
,
rhs
)
self
.
generate_generic_parallel_unpacking_code
(
code
,
rhs
,
use_loop
=
long_enough_for_a_loop
)
code
.
putln
(
"}"
)
for
value_node
in
self
.
coerced_unpacked_items
:
...
...
@@ -4920,9 +4924,16 @@ class SequenceNode(ExprNode):
self
.
args
[
i
].
generate_assignment_code
(
self
.
coerced_unpacked_items
[
i
],
code
)
def
generate_special_parallel_unpacking_code
(
self
,
code
,
sequence_type
):
def
generate_special_parallel_unpacking_code
(
self
,
code
,
sequence_type
,
use_loop
):
code
.
globalstate
.
use_utility_code
(
raise_need_more_values_to_unpack
)
code
.
globalstate
.
use_utility_code
(
raise_too_many_values_to_unpack
)
if
use_loop
:
# must be at the start of a C block!
code
.
putln
(
"PyObject** temps[%s] = {%s};"
%
(
len
(
self
.
unpacked_items
),
','
.
join
([
'&%s'
%
item
.
result
()
for
item
in
self
.
unpacked_items
])))
code
.
putln
(
"if (unlikely(Py%s_GET_SIZE(sequence) != %d)) {"
%
(
sequence_type
,
len
(
self
.
args
)))
code
.
putln
(
"if (Py%s_GET_SIZE(sequence) > %d) __Pyx_RaiseTooManyValuesError(%d);"
%
(
...
...
@@ -4930,14 +4941,35 @@ class SequenceNode(ExprNode):
code
.
putln
(
"else __Pyx_RaiseNeedMoreValuesError(Py%s_GET_SIZE(sequence));"
%
sequence_type
)
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
"}"
)
for
i
,
item
in
enumerate
(
self
.
unpacked_items
):
code
.
putln
(
"%s = Py%s_GET_ITEM(sequence, %d); "
%
(
item
.
result
(),
sequence_type
,
i
))
def
generate_generic_parallel_unpacking_code
(
self
,
code
,
rhs
):
if
use_loop
:
# shorter code in a loop works better for lists in CPython
counter
=
code
.
funcstate
.
allocate_temp
(
PyrexTypes
.
c_py_ssize_t_type
,
manage_ref
=
False
)
code
.
putln
(
"for (%s=0; %s < %s; %s++) {"
%
(
counter
,
counter
,
len
(
self
.
unpacked_items
),
counter
))
code
.
putln
(
"PyObject* item = Py%s_GET_ITEM(sequence, %s);"
%
(
sequence_type
,
counter
))
code
.
putln
(
"*(temps[%s]) = item;"
%
counter
)
code
.
put_incref
(
"item"
,
PyrexTypes
.
py_object_type
)
code
.
putln
(
"}"
)
code
.
funcstate
.
release_temp
(
counter
)
else
:
# unrolling the loop is very fast for tuples in CPython
for
i
,
item
in
enumerate
(
self
.
unpacked_items
):
code
.
putln
(
"%s = Py%s_GET_ITEM(sequence, %d); "
%
(
item
.
result
(),
sequence_type
,
i
))
code
.
put_incref
(
item
.
result
(),
item
.
ctype
())
def
generate_generic_parallel_unpacking_code
(
self
,
code
,
rhs
,
use_loop
):
code
.
globalstate
.
use_utility_code
(
iternext_unpacking_end_utility_code
)
code
.
globalstate
.
use_utility_code
(
raise_need_more_values_to_unpack
)
code
.
globalstate
.
use_utility_code
(
UtilityCode
.
load_cached
(
"IterFinish"
,
"ObjectHandling.c"
))
code
.
putln
(
"Py_ssize_t index = -1;"
)
# must be at the start of a C block!
if
use_loop
:
code
.
putln
(
"PyObject** temps[%s] = {%s};"
%
(
len
(
self
.
unpacked_items
),
','
.
join
([
'&%s'
%
item
.
result
()
for
item
in
self
.
unpacked_items
])))
iterator_temp
=
code
.
funcstate
.
allocate_temp
(
py_object_type
,
manage_ref
=
True
)
code
.
putln
(
"%s = PyObject_GetIter(%s); %s"
%
(
...
...
@@ -4952,32 +4984,37 @@ class SequenceNode(ExprNode):
iternext_func
,
iterator_temp
))
unpacking_error_label
=
code
.
new_label
(
'unpacking_failed'
)
code
.
use_label
(
unpacking_error_label
)
unpack_code
=
"%s(%s)"
%
(
iternext_func
,
iterator_temp
)
for
i
in
range
(
len
(
self
.
args
)):
item
=
self
.
unpacked_items
[
i
]
code
.
putln
(
"index = %d; %s = %s; if (unlikely(!%s)) goto %s;"
%
(
i
,
item
.
result
(),
typecast
(
item
.
ctype
(),
py_object_type
,
unpack_code
),
item
.
result
(),
unpacking_error_label
))
code
.
put_gotref
(
item
.
py_result
())
code
.
put_error_if_neg
(
self
.
pos
,
"__Pyx_IternextUnpackEndCheck(%s(%s), %d)"
%
(
iternext_func
,
iterator_temp
,
len
(
self
.
args
)))
if
use_loop
:
code
.
putln
(
"for (index=0; index < %s; index++) {"
%
len
(
self
.
unpacked_items
))
code
.
put
(
"PyObject* item = %s; if (unlikely(!item)) "
%
unpack_code
)
code
.
put_goto
(
unpacking_error_label
)
code
.
put_gotref
(
"item"
)
code
.
putln
(
"*(temps[index]) = item;"
)
code
.
putln
(
"}"
)
else
:
for
i
,
item
in
enumerate
(
self
.
unpacked_items
):
code
.
put
(
"index = %d; %s = %s; if (unlikely(!%s)) "
%
(
i
,
item
.
result
(),
unpack_code
,
item
.
result
()))
code
.
put_goto
(
unpacking_error_label
)
code
.
put_gotref
(
item
.
py_result
())
code
.
put_error_if_neg
(
self
.
pos
,
"__Pyx_IternextUnpackEndCheck(%s, %d)"
%
(
unpack_code
,
len
(
self
.
unpacked_items
)))
code
.
put_decref_clear
(
iterator_temp
,
py_object_type
)
code
.
funcstate
.
release_temp
(
iterator_temp
)
code
.
putln
(
"%s = NULL;"
%
iternext_func
)
code
.
funcstate
.
release_temp
(
iternext_func
)
unpacking_done_label
=
code
.
new_label
(
'unpacking_done'
)
code
.
put_goto
(
unpacking_done_label
)
code
.
put_label
(
unpacking_error_label
)
code
.
put_decref_clear
(
iterator_temp
,
py_object_type
)
code
.
putln
(
"if (PyErr_Occurred() && PyErr_ExceptionMatches(PyExc_StopIteration)) PyErr_Clear();"
)
code
.
putln
(
"if (!PyErr_Occurred()) __Pyx_RaiseNeedMoreValuesError(index);"
)
code
.
putln
(
"if (__Pyx_IterFinish() == 0) __Pyx_RaiseNeedMoreValuesError(index);"
)
code
.
putln
(
code
.
error_goto
(
self
.
pos
))
code
.
put_label
(
unpacking_done_label
)
...
...
tests/run/unpack.pyx
View file @
f98829fb
...
...
@@ -319,3 +319,54 @@ def failure_while_unpacking(it):
"""
a
,
b
,
c
=
it
return
a
,
b
,
c
def
unpack_many
(
it
):
"""
>>> items = range(1,13)
>>> unpack_many(items)
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many(iter(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many(list(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many(tuple(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
"""
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
=
it
return
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
def
unpack_many_tuple
(
tuple
it
):
"""
>>> items = range(1,13)
>>> unpack_many_tuple(tuple(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
"""
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
=
it
return
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
def
unpack_many_list
(
list
it
):
"""
>>> items = range(1,13)
>>> unpack_many_list(list(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
"""
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
=
it
return
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
def
unpack_many_int
(
it
):
"""
>>> items = range(1,13)
>>> unpack_many_int(items)
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many_int(iter(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many_int(list(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many_int(tuple(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
"""
cdef
int
b
cdef
long
f
cdef
Py_ssize_t
h
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
=
it
return
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
tests/run/unpack_fused.pyx
0 → 100644
View file @
f98829fb
ctypedef
fused
sequence
:
list
tuple
object
def
unpack_one
(
sequence
it
):
"""
>>> items = [1]
>>> unpack_one(items)
1
>>> unpack_one(iter(items))
1
>>> unpack_one(list(items))
1
>>> unpack_one(tuple(items))
1
"""
a
,
=
it
return
a
def
unpack_two
(
sequence
it
):
"""
>>> items = [1,2]
>>> unpack_two(items)
(1, 2)
>>> unpack_two(iter(items))
(1, 2)
>>> unpack_two(list(items))
(1, 2)
>>> unpack_two(tuple(items))
(1, 2)
"""
a
,
b
=
it
return
a
,
b
def
unpack_two_int
(
sequence
it
):
"""
>>> items = [1,2]
>>> unpack_two_int(items)
(1, 2)
>>> unpack_two_int(iter(items))
(1, 2)
>>> unpack_two_int(list(items))
(1, 2)
>>> unpack_two_int(tuple(items))
(1, 2)
>>> items = [1, object()]
>>> unpack_two_int(items)
Traceback (most recent call last):
TypeError: an integer is required
>>> unpack_two_int(iter(items))
Traceback (most recent call last):
TypeError: an integer is required
>>> unpack_two_int(list(items))
Traceback (most recent call last):
TypeError: an integer is required
>>> unpack_two_int(tuple(items))
Traceback (most recent call last):
TypeError: an integer is required
"""
cdef
int
b
a
,
b
=
it
return
a
,
b
def
unpack_many
(
sequence
it
):
"""
>>> items = range(1,13)
>>> unpack_many(items)
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many(iter(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many(list(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many(tuple(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
"""
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
=
it
return
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
def
unpack_many_int
(
sequence
it
):
"""
>>> items = range(1,13)
>>> unpack_many_int(items)
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many_int(iter(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many_int(list(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> unpack_many_int(tuple(items))
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
>>> items = range(1,10)
>>> unpack_many_int(items)
Traceback (most recent call last):
ValueError: need more than 9 values to unpack
>>> unpack_many_int(iter(items))
Traceback (most recent call last):
ValueError: need more than 9 values to unpack
>>> unpack_many_int(list(items))
Traceback (most recent call last):
ValueError: need more than 9 values to unpack
>>> unpack_many_int(tuple(items))
Traceback (most recent call last):
ValueError: need more than 9 values to unpack
"""
cdef
int
b
cdef
long
f
cdef
Py_ssize_t
h
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
=
it
return
a
,
b
,
c
,
d
,
e
,
f
,
g
,
h
,
i
,
j
,
k
,
l
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment