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
881a4c61
Commit
881a4c61
authored
Feb 11, 2010
by
Stefan Behnel
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
enable for-in iteration also for C arrays of known size
parent
4cf1611e
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
61 additions
and
29 deletions
+61
-29
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+7
-1
Cython/Compiler/Optimize.py
Cython/Compiler/Optimize.py
+26
-11
tests/run/carray_slicing.pyx
tests/run/carray_slicing.pyx
+28
-17
No files found.
Cython/Compiler/ExprNodes.py
View file @
881a4c61
...
@@ -1557,6 +1557,12 @@ class IteratorNode(ExprNode):
...
@@ -1557,6 +1557,12 @@ class IteratorNode(ExprNode):
def
analyse_types
(
self
,
env
):
def
analyse_types
(
self
,
env
):
self
.
sequence
.
analyse_types
(
env
)
self
.
sequence
.
analyse_types
(
env
)
if
isinstance
(
self
.
sequence
,
SliceIndexNode
)
and
\
(
self
.
sequence
.
base
.
type
.
is_array
or
self
.
sequence
.
base
.
type
.
is_ptr
)
\
or
self
.
sequence
.
type
.
is_array
and
self
.
sequence
.
type
.
size
is
not
None
:
# C array iteration will be transformed later on
pass
else
:
self
.
sequence
=
self
.
sequence
.
coerce_to_pyobject
(
env
)
self
.
sequence
=
self
.
sequence
.
coerce_to_pyobject
(
env
)
self
.
is_temp
=
1
self
.
is_temp
=
1
...
...
Cython/Compiler/Optimize.py
View file @
881a4c61
...
@@ -89,10 +89,12 @@ class IterationTransform(Visitor.VisitorTransform):
...
@@ -89,10 +89,12 @@ class IterationTransform(Visitor.VisitorTransform):
return
self
.
_transform_dict_iteration
(
return
self
.
_transform_dict_iteration
(
node
,
dict_obj
=
iterator
,
keys
=
True
,
values
=
False
)
node
,
dict_obj
=
iterator
,
keys
=
True
,
values
=
False
)
# C array
slice
iteration?
# C array
(slice)
iteration?
if
isinstance
(
iterator
,
ExprNodes
.
SliceIndexNode
)
and
\
if
isinstance
(
iterator
,
ExprNodes
.
SliceIndexNode
)
and
\
(
iterator
.
base
.
type
.
is_array
or
iterator
.
base
.
type
.
is_ptr
):
(
iterator
.
base
.
type
.
is_array
or
iterator
.
base
.
type
.
is_ptr
):
return
self
.
_transform_carray_iteration
(
node
,
iterator
)
return
self
.
_transform_carray_iteration
(
node
,
iterator
)
elif
iterator
.
type
.
is_array
:
return
self
.
_transform_carray_iteration
(
node
,
iterator
)
elif
not
isinstance
(
iterator
,
ExprNodes
.
SimpleCallNode
):
elif
not
isinstance
(
iterator
,
ExprNodes
.
SimpleCallNode
):
return
node
return
node
...
@@ -131,13 +133,26 @@ class IterationTransform(Visitor.VisitorTransform):
...
@@ -131,13 +133,26 @@ class IterationTransform(Visitor.VisitorTransform):
return
node
return
node
def
_transform_carray_iteration
(
self
,
node
,
slice_node
):
def
_transform_carray_iteration
(
self
,
node
,
slice_node
):
if
isinstance
(
slice_node
,
ExprNodes
.
SliceIndexNode
):
slice_base
=
slice_node
.
base
start
=
slice_node
.
start
start
=
slice_node
.
start
stop
=
slice_node
.
stop
stop
=
slice_node
.
stop
step
=
None
step
=
None
if
not
stop
:
if
not
stop
:
return
node
return
node
elif
slice_node
.
type
.
is_array
and
slice_node
.
type
.
size
is
not
None
:
slice_base
=
slice_node
start
=
None
stop
=
ExprNodes
.
IntNode
(
slice_node
.
pos
,
value
=
str
(
slice_node
.
type
.
size
))
step
=
None
else
:
return
node
carray_ptr
=
slice_node
.
base
.
coerce_to_simple
(
self
.
current_scope
)
ptr_type
=
slice_base
.
type
if
ptr_type
.
is_array
:
ptr_type
=
ptr_type
.
element_ptr_type
()
carray_ptr
=
slice_base
.
coerce_to_simple
(
self
.
current_scope
)
if
start
and
start
.
constant_result
!=
0
:
if
start
and
start
.
constant_result
!=
0
:
start_ptr_node
=
ExprNodes
.
AddNode
(
start_ptr_node
=
ExprNodes
.
AddNode
(
...
@@ -145,7 +160,7 @@ class IterationTransform(Visitor.VisitorTransform):
...
@@ -145,7 +160,7 @@ class IterationTransform(Visitor.VisitorTransform):
operand1
=
carray_ptr
,
operand1
=
carray_ptr
,
operator
=
'+'
,
operator
=
'+'
,
operand2
=
start
,
operand2
=
start
,
type
=
carray_ptr
.
type
)
type
=
ptr_
type
)
else
:
else
:
start_ptr_node
=
carray_ptr
start_ptr_node
=
carray_ptr
...
@@ -154,13 +169,13 @@ class IterationTransform(Visitor.VisitorTransform):
...
@@ -154,13 +169,13 @@ class IterationTransform(Visitor.VisitorTransform):
operand1
=
carray_ptr
,
operand1
=
carray_ptr
,
operator
=
'+'
,
operator
=
'+'
,
operand2
=
stop
,
operand2
=
stop
,
type
=
carray_ptr
.
type
type
=
ptr_
type
).
coerce_to_simple
(
self
.
current_scope
)
).
coerce_to_simple
(
self
.
current_scope
)
counter
=
UtilNodes
.
TempHandle
(
carray_ptr
.
type
)
counter
=
UtilNodes
.
TempHandle
(
ptr_
type
)
counter_temp
=
counter
.
ref
(
node
.
target
.
pos
)
counter_temp
=
counter
.
ref
(
node
.
target
.
pos
)
if
slice_
node
.
base
.
type
.
is_string
and
node
.
target
.
type
.
is_pyobject
:
if
slice_base
.
type
.
is_string
and
node
.
target
.
type
.
is_pyobject
:
# special case: char* -> bytes
# special case: char* -> bytes
target_value
=
ExprNodes
.
SliceIndexNode
(
target_value
=
ExprNodes
.
SliceIndexNode
(
node
.
target
.
pos
,
node
.
target
.
pos
,
...
@@ -181,7 +196,7 @@ class IterationTransform(Visitor.VisitorTransform):
...
@@ -181,7 +196,7 @@ class IterationTransform(Visitor.VisitorTransform):
type
=
PyrexTypes
.
c_int_type
),
type
=
PyrexTypes
.
c_int_type
),
base
=
counter_temp
,
base
=
counter_temp
,
is_buffer_access
=
False
,
is_buffer_access
=
False
,
type
=
carray_ptr
.
type
.
base_type
)
type
=
ptr_
type
.
base_type
)
if
target_value
.
type
!=
node
.
target
.
type
:
if
target_value
.
type
!=
node
.
target
.
type
:
target_value
=
target_value
.
coerce_to
(
node
.
target
.
type
,
target_value
=
target_value
.
coerce_to
(
node
.
target
.
type
,
...
...
tests/run/carray_slicing.pyx
View file @
881a4c61
...
@@ -111,21 +111,32 @@ def slice_charptr_for_loop_c_enumerate():
...
@@ -111,21 +111,32 @@ def slice_charptr_for_loop_c_enumerate():
############################################################
############################################################
# tests for int* slicing
# tests for int* slicing
##
cdef int cints[6]
cdef
int
cints
[
6
]
##
for i in range(6):
for
i
in
range
(
6
):
##
cints[i] = i
cints
[
i
]
=
i
## @cython.test_assert_path_exists("//ForFromStatNode",
@
cython
.
test_assert_path_exists
(
"//ForFromStatNode"
,
## "//ForFromStatNode//IndexNode")
"//ForFromStatNode//IndexNode"
)
## @cython.test_fail_if_path_exists("//ForInStatNode")
@
cython
.
test_fail_if_path_exists
(
"//ForInStatNode"
)
## def slice_intptr_for_loop_c():
def
slice_intptr_for_loop_c
():
## """
"""
## >>> slice_intptr_for_loop_c()
>>> slice_intptr_for_loop_c()
## [0, 1, 2]
[0, 1, 2]
## [1, 2, 3, 4]
[1, 2, 3, 4]
## [4, 5]
[4, 5]
## """
"""
## cdef int i
cdef
int
i
## print [ i for i in cints[:3] ]
print
[
i
for
i
in
cints
[:
3
]
]
## print [ i for i in cints[1:5] ]
print
[
i
for
i
in
cints
[
1
:
5
]
]
## print [ i for i in cints[4:6] ]
print
[
i
for
i
in
cints
[
4
:
6
]
]
@
cython
.
test_assert_path_exists
(
"//ForFromStatNode"
,
"//ForFromStatNode//IndexNode"
)
@
cython
.
test_fail_if_path_exists
(
"//ForInStatNode"
)
def
iter_intptr_for_loop_c
():
"""
>>> iter_intptr_for_loop_c()
[0, 1, 2, 3, 4, 5]
"""
cdef
int
i
print
[
i
for
i
in
cints
]
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