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
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Boxiang Sun
cython
Commits
656196c3
Commit
656196c3
authored
Oct 11, 2013
by
Stefan Behnel
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
reduce Unicode iteration over Latin-1 literals to bytes iteration
parent
5f0b23cb
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
65 additions
and
15 deletions
+65
-15
Cython/Compiler/Optimize.py
Cython/Compiler/Optimize.py
+45
-12
tests/run/cython3.pyx
tests/run/cython3.pyx
+1
-1
tests/run/py_ucs4_type.pyx
tests/run/py_ucs4_type.pyx
+19
-2
No files found.
Cython/Compiler/Optimize.py
View file @
656196c3
...
@@ -285,6 +285,29 @@ class IterationTransform(Visitor.EnvTransform):
...
@@ -285,6 +285,29 @@ class IterationTransform(Visitor.EnvTransform):
exception_value
=
'-1'
)
exception_value
=
'-1'
)
def
_transform_unicode_iteration
(
self
,
node
,
slice_node
,
reversed
=
False
):
def
_transform_unicode_iteration
(
self
,
node
,
slice_node
,
reversed
=
False
):
if
slice_node
.
is_literal
:
# try to reduce to byte iteration for plain Latin-1 strings
try
:
bytes_value
=
BytesLiteral
(
slice_node
.
value
.
encode
(
'latin1'
))
except
UnicodeEncodeError
:
pass
else
:
bytes_slice
=
ExprNodes
.
SliceIndexNode
(
slice_node
.
pos
,
base
=
ExprNodes
.
BytesNode
(
slice_node
.
pos
,
value
=
bytes_value
,
constant_result
=
bytes_value
,
type
=
PyrexTypes
.
c_char_ptr_type
).
coerce_to
(
PyrexTypes
.
c_uchar_ptr_type
,
self
.
current_env
()),
start
=
None
,
stop
=
ExprNodes
.
IntNode
(
slice_node
.
pos
,
value
=
len
(
bytes_value
),
constant_result
=
len
(
bytes_value
),
type
=
PyrexTypes
.
c_py_ssize_t_type
),
type
=
Builtin
.
unicode_type
,
# hint for Python conversion
)
return
self
.
_transform_carray_iteration
(
node
,
bytes_slice
,
reversed
)
unpack_temp_node
=
UtilNodes
.
LetRefNode
(
unpack_temp_node
=
UtilNodes
.
LetRefNode
(
slice_node
.
as_none_safe_node
(
"'NoneType' is not iterable"
))
slice_node
.
as_none_safe_node
(
"'NoneType' is not iterable"
))
...
@@ -455,7 +478,16 @@ class IterationTransform(Visitor.EnvTransform):
...
@@ -455,7 +478,16 @@ class IterationTransform(Visitor.EnvTransform):
counter_temp
=
counter
.
ref
(
node
.
target
.
pos
)
counter_temp
=
counter
.
ref
(
node
.
target
.
pos
)
if
slice_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/unicode
if
slice_node
.
type
is
Builtin
.
unicode_type
:
target_value
=
ExprNodes
.
CastNode
(
ExprNodes
.
DereferenceNode
(
node
.
target
.
pos
,
operand
=
counter_temp
,
type
=
ptr_type
.
base_type
),
PyrexTypes
.
c_py_ucs4_type
).
coerce_to
(
node
.
target
.
type
,
self
.
current_env
())
else
:
# char* -> bytes coercion requires slicing, not indexing
target_value
=
ExprNodes
.
SliceIndexNode
(
target_value
=
ExprNodes
.
SliceIndexNode
(
node
.
target
.
pos
,
node
.
target
.
pos
,
start
=
ExprNodes
.
IntNode
(
node
.
target
.
pos
,
value
=
'0'
,
start
=
ExprNodes
.
IntNode
(
node
.
target
.
pos
,
value
=
'0'
,
...
@@ -471,6 +503,7 @@ class IterationTransform(Visitor.EnvTransform):
...
@@ -471,6 +503,7 @@ class IterationTransform(Visitor.EnvTransform):
# Allow iteration with pointer target to avoid copy.
# Allow iteration with pointer target to avoid copy.
target_value
=
counter_temp
target_value
=
counter_temp
else
:
else
:
# TODO: can this safely be replaced with DereferenceNode() as above?
target_value
=
ExprNodes
.
IndexNode
(
target_value
=
ExprNodes
.
IndexNode
(
node
.
target
.
pos
,
node
.
target
.
pos
,
index
=
ExprNodes
.
IntNode
(
node
.
target
.
pos
,
value
=
'0'
,
index
=
ExprNodes
.
IntNode
(
node
.
target
.
pos
,
value
=
'0'
,
...
...
tests/run/cython3.pyx
View file @
656196c3
...
@@ -291,7 +291,7 @@ def loop_over_unicode_literal():
...
@@ -291,7 +291,7 @@ def loop_over_unicode_literal():
"""
"""
# Py_UCS4 can represent any Unicode character
# Py_UCS4 can represent any Unicode character
for
uchar
in
'abcdefg'
:
for
uchar
in
'abcdefg'
:
pass
assert
uchar
in
'abcdefg'
return
cython
.
typeof
(
uchar
)
return
cython
.
typeof
(
uchar
)
def
list_comp
():
def
list_comp
():
...
...
tests/run/py_ucs4_type.pyx
View file @
656196c3
...
@@ -209,14 +209,31 @@ def count_lower_case_characters_slice_reversed(unicode ustring):
...
@@ -209,14 +209,31 @@ def count_lower_case_characters_slice_reversed(unicode ustring):
count
+=
1
count
+=
1
return
count
return
count
def
loop_object_over_latin1_unicode_literal
():
"""
>>> result = loop_object_over_latin1_unicode_literal()
>>> print(result[:-1])
abcdefg
>>> ord(result[-1]) == 0xD7
True
"""
cdef
object
uchar
chars
=
[]
for
uchar
in
u'abcdefg
\
xD7
'
:
chars
.
append
(
uchar
)
return
u''
.
join
(
chars
)
def
loop_object_over_unicode_literal
():
def
loop_object_over_unicode_literal
():
"""
"""
>>> print(loop_object_over_unicode_literal())
>>> result = loop_object_over_unicode_literal()
>>> print(result[:-1])
abcdefg
abcdefg
>>> ord(result[-1]) == 0xF8FD
True
"""
"""
cdef
object
uchar
cdef
object
uchar
chars
=
[]
chars
=
[]
for
uchar
in
u'abcdefg'
:
for
uchar
in
u'abcdefg
\
uF8FD
'
:
chars
.
append
(
uchar
)
chars
.
append
(
uchar
)
return
u''
.
join
(
chars
)
return
u''
.
join
(
chars
)
...
...
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