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
Gwenaël Samain
cython
Commits
ffb006d7
Commit
ffb006d7
authored
Feb 23, 2013
by
Stefan Behnel
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
refactor constant string slicing and guard it against platform specific unicode string length
parent
d1cc779e
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
88 additions
and
27 deletions
+88
-27
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+58
-15
Cython/Compiler/Optimize.py
Cython/Compiler/Optimize.py
+3
-12
tests/run/constant_folding_cy.pyx
tests/run/constant_folding_cy.pyx
+27
-0
No files found.
Cython/Compiler/ExprNodes.py
View file @
ffb006d7
...
...
@@ -1071,6 +1071,12 @@ class BytesNode(ConstNode):
def
calculate_constant_result
(
self
):
self
.
constant_result
=
self
.
value
def
as_sliced_node
(
self
,
start
,
stop
,
step
=
None
):
value
=
StringEncoding
.
BytesLiteral
(
self
.
value
[
start
:
stop
:
step
])
value
.
encoding
=
self
.
value
.
encoding
return
BytesNode
(
self
.
pos
,
value
=
value
,
constant_result
=
value
)
def
compile_time_value
(
self
,
denv
):
return
self
.
value
...
...
@@ -1155,6 +1161,22 @@ class UnicodeNode(PyConstNode):
def
calculate_constant_result
(
self
):
self
.
constant_result
=
self
.
value
def
as_sliced_node
(
self
,
start
,
stop
,
step
=
None
):
if
_string_contains_surrogates
(
self
.
value
[:
stop
]):
# this is unsafe as it may give different results in different runtimes
return
None
value
=
StringEncoding
.
EncodedString
(
self
.
value
[
start
:
stop
:
step
])
value
.
encoding
=
self
.
value
.
encoding
if
self
.
bytes_value
is
not
None
:
bytes_value
=
StringEncoding
.
BytesLiteral
(
self
.
bytes_value
[
start
:
stop
:
step
])
bytes_value
.
encoding
=
self
.
bytes_value
.
encoding
else
:
bytes_value
=
None
return
UnicodeNode
(
self
.
pos
,
value
=
value
,
bytes_value
=
bytes_value
,
constant_result
=
value
)
def
coerce_to
(
self
,
dst_type
,
env
):
if
dst_type
is
self
.
type
:
pass
...
...
@@ -1181,21 +1203,7 @@ class UnicodeNode(PyConstNode):
## and (0xDC00 <= self.value[1] <= 0xDFFF))
def
contains_surrogates
(
self
):
# Check if the unicode string contains surrogate code points
# on a CPython platform with wide (UCS-4) or narrow (UTF-16)
# Unicode, i.e. characters that would be spelled as two
# separate code units on a narrow platform.
for
c
in
map
(
ord
,
self
.
value
):
if
c
>
65535
:
# can only happen on wide platforms
return
True
# We only look for the first code unit (D800-DBFF) of a
# surrogate pair - if we find one, the other one
# (DC00-DFFF) is likely there, too. If we don't find it,
# any second code unit cannot make for a surrogate pair by
# itself.
if
0xD800
<=
c
<=
0xDBFF
:
return
True
return
False
return
_string_contains_surrogates
(
self
.
value
)
def
generate_evaluation_code
(
self
,
code
):
self
.
result_code
=
code
.
get_py_string_const
(
self
.
value
)
...
...
@@ -1223,6 +1231,21 @@ class StringNode(PyConstNode):
def
calculate_constant_result
(
self
):
self
.
constant_result
=
self
.
value
def
as_sliced_node
(
self
,
start
,
stop
,
step
=
None
):
value
=
type
(
self
.
value
)(
self
.
value
[
start
:
stop
:
step
])
value
.
encoding
=
self
.
value
.
encoding
if
self
.
unicode_value
is
not
None
:
if
_string_contains_surrogates
(
self
.
unicode_value
[:
stop
]):
# this is unsafe as it may give different results in different runtimes
return
None
unicode_value
=
StringEncoding
.
EncodedString
(
self
.
unicode_value
[
start
:
stop
:
step
])
else
:
unicode_value
=
None
return
StringNode
(
self
.
pos
,
value
=
value
,
unicode_value
=
unicode_value
,
constant_result
=
value
,
is_identifier
=
self
.
is_identifier
)
def
coerce_to
(
self
,
dst_type
,
env
):
if
dst_type
is
not
py_object_type
and
not
str_type
.
subtype_of
(
dst_type
):
# if dst_type is Builtin.bytes_type:
...
...
@@ -1257,6 +1280,26 @@ class IdentifierStringNode(StringNode):
is_identifier
=
True
def
_string_contains_surrogates
(
ustring
):
"""
Check if the unicode string contains surrogate code points
on a CPython platform with wide (UCS-4) or narrow (UTF-16)
Unicode, i.e. characters that would be spelled as two
separate code units on a narrow platform.
"""
for
c
in
map
(
ord
,
ustring
):
if
c
>
65535
:
# can only happen on wide platforms
return
True
# We only look for the first code unit (D800-DBFF) of a
# surrogate pair - if we find one, the other one
# (DC00-DFFF) is likely there, too. If we don't find it,
# any second code unit cannot make for a surrogate pair by
# itself.
if
0xD800
<=
c
<=
0xDBFF
:
return
True
return
False
class
ImagNode
(
AtomicExprNode
):
# Imaginary number literal
#
...
...
Cython/Compiler/Optimize.py
View file @
ffb006d7
...
...
@@ -3206,18 +3206,9 @@ class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations):
base
.
args
=
base
.
args
[
start
:
stop
]
return
base
elif
base
.
is_string_literal
:
value
=
type
(
base
.
value
)(
node
.
constant_result
)
value
.
encoding
=
base
.
value
.
encoding
base
.
value
=
value
if
isinstance
(
base
,
ExprNodes
.
StringNode
):
if
base
.
unicode_value
is
not
None
:
base
.
unicode_value
=
EncodedString
(
base
.
unicode_value
[
start
:
stop
])
elif
isinstance
(
base
,
ExprNodes
.
UnicodeNode
):
if
base
.
bytes_value
is
not
None
:
base
.
bytes_value
=
BytesLiteral
(
base
.
bytes_value
[
start
:
stop
])
return
base
base
=
base
.
as_sliced_node
(
start
,
stop
)
if
base
is
not
None
:
return
base
return
node
def
visit_ForInStatNode
(
self
,
node
):
...
...
tests/run/constant_folding_cy.pyx
View file @
ffb006d7
...
...
@@ -7,6 +7,7 @@ cimport cython
bstring
=
b'abc
\
xE9
def'
ustring
=
u'abc
\
xE9
def'
surrogates_ustring
=
u'abc
\
U00010000
def'
@
cython
.
test_fail_if_path_exists
(
...
...
@@ -53,3 +54,29 @@ def unicode_slicing2():
str3
=
u'abc
\
xE9
def'
[
2
:
4
]
return
str0
,
str1
,
str2
,
str3
@
cython
.
test_assert_path_exists
(
"//SliceIndexNode"
,
)
def
unicode_slicing_unsafe_surrogates2
():
"""
>>> unicode_slicing_unsafe_surrogates2() == surrogates_ustring[2:]
True
"""
ustring
=
u'abc
\
U00010000
def'
[
2
:]
return
ustring
@
cython
.
test_fail_if_path_exists
(
"//SliceIndexNode"
,
)
def
unicode_slicing_safe_surrogates2
():
"""
>>> unicode_slicing_safe_surrogates2() == surrogates_ustring[:2]
True
>>> print(unicode_slicing_safe_surrogates2())
ab
"""
ustring
=
u'abc
\
U00010000
def'
[:
2
]
return
ustring
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