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
Kirill Smelkov
cython
Commits
b69b4a5b
Commit
b69b4a5b
authored
May 11, 2012
by
Robert Bradshaw
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
cythonize performance improvements for large codebases
parent
9560cbec
Changes
5
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
236 additions
and
161 deletions
+236
-161
Cython/Build/Dependencies.py
Cython/Build/Dependencies.py
+85
-84
Cython/Build/Tests/TestStripLiterals.py
Cython/Build/Tests/TestStripLiterals.py
+2
-2
Cython/Compiler/Main.py
Cython/Compiler/Main.py
+5
-62
Cython/Utils.py
Cython/Utils.py
+143
-13
setup.py
setup.py
+1
-0
No files found.
Cython/Build/Dependencies.py
View file @
b69b4a5b
This diff is collapsed.
Click to expand it.
Cython/Build/Tests/TestStripLiterals.py
View file @
b69b4a5b
...
@@ -35,10 +35,10 @@ class TestStripLiterals(CythonTest):
...
@@ -35,10 +35,10 @@ class TestStripLiterals(CythonTest):
self
.
t
(
"u'abc'"
,
"u'_L1_'"
)
self
.
t
(
"u'abc'"
,
"u'_L1_'"
)
def
test_raw
(
self
):
def
test_raw
(
self
):
self
.
t
(
r"r'abc\'"
,
"r'_L1_'"
)
self
.
t
(
r"r'abc\
\
'"
,
"r'_L1_'"
)
def
test_raw_unicode
(
self
):
def
test_raw_unicode
(
self
):
self
.
t
(
r"ru'abc\'"
,
"ru'_L1_'"
)
self
.
t
(
r"ru'abc\
\
'"
,
"ru'_L1_'"
)
def
test_comment
(
self
):
def
test_comment
(
self
):
self
.
t
(
"abc # foo"
,
"abc #_L1_"
)
self
.
t
(
"abc # foo"
,
"abc #_L1_"
)
...
...
Cython/Compiler/Main.py
View file @
b69b4a5b
...
@@ -222,65 +222,14 @@ class Context(object):
...
@@ -222,65 +222,14 @@ class Context(object):
def
search_include_directories
(
self
,
qualified_name
,
suffix
,
pos
,
def
search_include_directories
(
self
,
qualified_name
,
suffix
,
pos
,
include
=
False
,
sys_path
=
False
):
include
=
False
,
sys_path
=
False
):
# Search the list of include directories for the given
return
Utils
.
search_include_directories
(
# file name. If a source file position is given, first
tuple
(
self
.
include_directories
),
qualified_name
,
suffix
,
pos
,
include
,
sys_path
)
# searches the directory containing that file. Returns
# None if not found, but does not report an error.
# The 'include' option will disable package dereferencing.
# If 'sys_path' is True, also search sys.path.
dirs
=
self
.
include_directories
if
sys_path
:
dirs
=
dirs
+
sys
.
path
if
pos
:
file_desc
=
pos
[
0
]
if
not
isinstance
(
file_desc
,
FileSourceDescriptor
):
raise
RuntimeError
(
"Only file sources for code supported"
)
if
include
:
dirs
=
[
os
.
path
.
dirname
(
file_desc
.
filename
)]
+
dirs
else
:
dirs
=
[
self
.
find_root_package_dir
(
file_desc
.
filename
)]
+
dirs
dotted_filename
=
qualified_name
if
suffix
:
dotted_filename
+=
suffix
if
not
include
:
names
=
qualified_name
.
split
(
'.'
)
package_names
=
names
[:
-
1
]
module_name
=
names
[
-
1
]
module_filename
=
module_name
+
suffix
package_filename
=
"__init__"
+
suffix
for
dir
in
dirs
:
path
=
os
.
path
.
join
(
dir
,
dotted_filename
)
if
Utils
.
path_exists
(
path
):
return
path
if
not
include
:
package_dir
=
self
.
check_package_dir
(
dir
,
package_names
)
if
package_dir
is
not
None
:
path
=
os
.
path
.
join
(
package_dir
,
module_filename
)
if
Utils
.
path_exists
(
path
):
return
path
path
=
os
.
path
.
join
(
dir
,
package_dir
,
module_name
,
package_filename
)
if
Utils
.
path_exists
(
path
):
return
path
return
None
def
find_root_package_dir
(
self
,
file_path
):
def
find_root_package_dir
(
self
,
file_path
):
dir
=
os
.
path
.
dirname
(
file_path
)
return
Utils
.
find_root_package_dir
(
file_path
)
while
self
.
is_package_dir
(
dir
):
parent
=
os
.
path
.
dirname
(
dir
)
if
parent
==
dir
:
break
dir
=
parent
return
dir
def
check_package_dir
(
self
,
dir
,
package_names
):
def
check_package_dir
(
self
,
dir
,
package_names
):
for
dirname
in
package_names
:
return
Utils
.
check_package_dir
(
dir
,
tuple
(
package_names
))
dir
=
os
.
path
.
join
(
dir
,
dirname
)
if
not
self
.
is_package_dir
(
dir
):
return
None
return
dir
def
c_file_out_of_date
(
self
,
source_path
):
def
c_file_out_of_date
(
self
,
source_path
):
c_path
=
Utils
.
replace_suffix
(
source_path
,
".c"
)
c_path
=
Utils
.
replace_suffix
(
source_path
,
".c"
)
...
@@ -309,13 +258,7 @@ class Context(object):
...
@@ -309,13 +258,7 @@ class Context(object):
if
kind
==
"cimport"
]
if
kind
==
"cimport"
]
def
is_package_dir
(
self
,
dir_path
):
def
is_package_dir
(
self
,
dir_path
):
# Return true if the given directory is a package directory.
return
Utils
.
is_package_dir
(
dir_path
)
for
filename
in
(
"__init__.py"
,
"__init__.pyx"
,
"__init__.pxd"
):
path
=
os
.
path
.
join
(
dir_path
,
filename
)
if
Utils
.
path_exists
(
path
):
return
1
def
read_dependency_file
(
self
,
source_path
):
def
read_dependency_file
(
self
,
source_path
):
dep_path
=
Utils
.
replace_suffix
(
source_path
,
".dep"
)
dep_path
=
Utils
.
replace_suffix
(
source_path
,
".dep"
)
...
...
Cython/Utils.py
View file @
b69b4a5b
...
@@ -7,6 +7,29 @@ import os, sys, re, codecs
...
@@ -7,6 +7,29 @@ import os, sys, re, codecs
modification_time
=
os
.
path
.
getmtime
modification_time
=
os
.
path
.
getmtime
def
cached_function
(
f
):
cache
=
{}
uncomputed
=
object
()
def
wrapper
(
*
args
):
res
=
cache
.
get
(
args
,
uncomputed
)
if
res
is
uncomputed
:
res
=
cache
[
args
]
=
f
(
*
args
)
return
res
return
wrapper
def
cached_method
(
f
):
cache_name
=
'__%s_cache'
%
f
.
__name__
def
wrapper
(
self
,
*
args
):
cache
=
getattr
(
self
,
cache_name
,
None
)
if
cache
is
None
:
cache
=
{}
setattr
(
self
,
cache_name
,
cache
)
if
args
in
cache
:
return
cache
[
args
]
res
=
cache
[
args
]
=
f
(
self
,
*
args
)
return
res
return
wrapper
def
replace_suffix
(
path
,
newsuf
):
def
replace_suffix
(
path
,
newsuf
):
base
,
_
=
os
.
path
.
splitext
(
path
)
base
,
_
=
os
.
path
.
splitext
(
path
)
return
base
+
newsuf
return
base
+
newsuf
...
@@ -43,6 +66,82 @@ def file_newer_than(path, time):
...
@@ -43,6 +66,82 @@ def file_newer_than(path, time):
ftime
=
modification_time
(
path
)
ftime
=
modification_time
(
path
)
return
ftime
>
time
return
ftime
>
time
@
cached_function
def
search_include_directories
(
dirs
,
qualified_name
,
suffix
,
pos
,
include
=
False
,
sys_path
=
False
):
# Search the list of include directories for the given
# file name. If a source file position is given, first
# searches the directory containing that file. Returns
# None if not found, but does not report an error.
# The 'include' option will disable package dereferencing.
# If 'sys_path' is True, also search sys.path.
if
sys_path
:
dirs
=
dirs
+
tuple
(
sys
.
path
)
if
pos
:
file_desc
=
pos
[
0
]
from
Cython.Compiler.Scanning
import
FileSourceDescriptor
if
not
isinstance
(
file_desc
,
FileSourceDescriptor
):
raise
RuntimeError
(
"Only file sources for code supported"
)
if
include
:
dirs
=
(
os
.
path
.
dirname
(
file_desc
.
filename
),)
+
dirs
else
:
dirs
=
(
find_root_package_dir
(
file_desc
.
filename
),)
+
dirs
dotted_filename
=
qualified_name
if
suffix
:
dotted_filename
+=
suffix
if
not
include
:
names
=
qualified_name
.
split
(
'.'
)
package_names
=
tuple
(
names
[:
-
1
])
module_name
=
names
[
-
1
]
module_filename
=
module_name
+
suffix
package_filename
=
"__init__"
+
suffix
for
dir
in
dirs
:
path
=
os
.
path
.
join
(
dir
,
dotted_filename
)
if
path_exists
(
path
):
return
path
if
not
include
:
package_dir
=
check_package_dir
(
dir
,
package_names
)
if
package_dir
is
not
None
:
path
=
os
.
path
.
join
(
package_dir
,
module_filename
)
if
path_exists
(
path
):
return
path
path
=
os
.
path
.
join
(
dir
,
package_dir
,
module_name
,
package_filename
)
if
path_exists
(
path
):
return
path
return
None
@
cached_function
def
find_root_package_dir
(
file_path
):
dir
=
os
.
path
.
dirname
(
file_path
)
while
is_package_dir
(
dir
):
parent
=
os
.
path
.
dirname
(
dir
)
if
parent
==
dir
:
break
dir
=
parent
return
dir
@
cached_function
def
check_package_dir
(
dir
,
package_names
):
for
dirname
in
package_names
:
dir
=
os
.
path
.
join
(
dir
,
dirname
)
if
not
is_package_dir
(
dir
):
return
None
return
dir
@
cached_function
def
is_package_dir
(
dir_path
):
for
filename
in
(
"__init__.py"
,
"__init__.pyx"
,
"__init__.pxd"
):
path
=
os
.
path
.
join
(
dir_path
,
filename
)
if
path_exists
(
path
):
return
1
@
cached_function
def
path_exists
(
path
):
def
path_exists
(
path
):
# try on the filesystem first
# try on the filesystem first
if
os
.
path
.
exists
(
path
):
if
os
.
path
.
exists
(
path
):
...
@@ -85,9 +184,26 @@ def decode_filename(filename):
...
@@ -85,9 +184,26 @@ def decode_filename(filename):
_match_file_encoding
=
re
.
compile
(
u"coding[:=]
\
s*([-
\
w.]+)"
).
search
_match_file_encoding
=
re
.
compile
(
u"coding[:=]
\
s*([-
\
w.]+)"
).
search
def
detect_file_encoding
(
source_filename
):
def
detect_file_encoding
(
source_filename
):
# PEPs 263 and 3120
f
=
open_source_file
(
source_filename
,
encoding
=
"UTF-8"
,
error_handling
=
'ignore'
)
f
=
open_source_file
(
source_filename
,
encoding
=
"UTF-8"
,
error_handling
=
'ignore'
)
try
:
try
:
return
detect_opened_file_encoding
(
f
)
finally
:
f
.
close
()
def
detect_opened_file_encoding
(
f
):
# PEPs 263 and 3120
# Most of the time the first two lines fall in the first 250 chars,
# and this bulk read/split is much faster.
lines
=
f
.
read
(
250
).
split
(
"
\
n
"
)
if
len
(
lines
)
>
2
:
m
=
_match_file_encoding
(
lines
[
0
])
or
_match_file_encoding
(
lines
[
1
])
if
m
:
return
m
.
group
(
1
)
else
:
return
"UTF-8"
else
:
# Fallback to one-char-at-a-time detection.
f
.
seek
(
0
)
chars
=
[]
chars
=
[]
for
i
in
range
(
2
):
for
i
in
range
(
2
):
c
=
f
.
read
(
1
)
c
=
f
.
read
(
1
)
...
@@ -97,8 +213,6 @@ def detect_file_encoding(source_filename):
...
@@ -97,8 +213,6 @@ def detect_file_encoding(source_filename):
encoding
=
_match_file_encoding
(
u''
.
join
(
chars
))
encoding
=
_match_file_encoding
(
u''
.
join
(
chars
))
if
encoding
:
if
encoding
:
return
encoding
.
group
(
1
)
return
encoding
.
group
(
1
)
finally
:
f
.
close
()
return
"UTF-8"
return
"UTF-8"
normalise_newlines
=
re
.
compile
(
u'
\
r
\
n
?|
\
n
'
).
sub
normalise_newlines
=
re
.
compile
(
u'
\
r
\
n
?|
\
n
'
).
sub
...
@@ -111,6 +225,7 @@ class NormalisedNewlineStream(object):
...
@@ -111,6 +225,7 @@ class NormalisedNewlineStream(object):
"""
"""
def
__init__
(
self
,
stream
):
def
__init__
(
self
,
stream
):
# let's assume .read() doesn't change
# let's assume .read() doesn't change
self
.
stream
=
stream
self
.
_read
=
stream
.
read
self
.
_read
=
stream
.
read
self
.
close
=
stream
.
close
self
.
close
=
stream
.
close
self
.
encoding
=
getattr
(
stream
,
'encoding'
,
'UTF-8'
)
self
.
encoding
=
getattr
(
stream
,
'encoding'
,
'UTF-8'
)
...
@@ -133,6 +248,12 @@ class NormalisedNewlineStream(object):
...
@@ -133,6 +248,12 @@ class NormalisedNewlineStream(object):
return
u''
.
join
(
content
).
splitlines
(
True
)
return
u''
.
join
(
content
).
splitlines
(
True
)
def
seek
(
self
,
pos
):
if
pos
==
0
:
self
.
stream
.
seek
(
0
)
else
:
raise
NotImplementedError
io
=
None
io
=
None
if
sys
.
version_info
>=
(
2
,
6
):
if
sys
.
version_info
>=
(
2
,
6
):
try
:
try
:
...
@@ -144,17 +265,26 @@ def open_source_file(source_filename, mode="r",
...
@@ -144,17 +265,26 @@ def open_source_file(source_filename, mode="r",
encoding
=
None
,
error_handling
=
None
,
encoding
=
None
,
error_handling
=
None
,
require_normalised_newlines
=
True
):
require_normalised_newlines
=
True
):
if
encoding
is
None
:
if
encoding
is
None
:
encoding
=
detect_file_encoding
(
source_filename
)
# Most of the time the coding is unspecified, so be optimistic that
# it's UTF-8.
f
=
open_source_file
(
source_filename
,
encoding
=
"UTF-8"
,
mode
=
mode
,
error_handling
=
'ignore'
)
encoding
=
detect_opened_file_encoding
(
f
)
if
encoding
==
"UTF-8"
and
error_handling
==
'ignore'
and
require_normalised_newlines
:
f
.
seek
(
0
)
return
f
else
:
f
.
close
()
#
#
try
:
if
not
os
.
path
.
exists
(
source_filename
):
loader
=
__loader__
try
:
if
source_filename
.
startswith
(
loader
.
archive
):
loader
=
__loader__
return
open_source_from_loader
(
if
source_filename
.
startswith
(
loader
.
archive
):
loader
,
source_filename
,
return
open_source_from_loader
(
encoding
,
error_handling
,
loader
,
source_filename
,
require_normalised_newlines
)
encoding
,
error_handling
,
except
(
NameError
,
AttributeError
):
require_normalised_newlines
)
pass
except
(
NameError
,
AttributeError
):
pass
#
#
if
io
is
not
None
:
if
io
is
not
None
:
return
io
.
open
(
source_filename
,
mode
=
mode
,
return
io
.
open
(
source_filename
,
mode
=
mode
,
...
...
setup.py
View file @
b69b4a5b
...
@@ -116,6 +116,7 @@ def compile_cython_modules(profile=False, compile_more=False, cython_with_refnan
...
@@ -116,6 +116,7 @@ def compile_cython_modules(profile=False, compile_more=False, cython_with_refnan
]
]
if
compile_more
:
if
compile_more
:
compiled_modules
.
extend
([
compiled_modules
.
extend
([
"Cython.Build.Dependencies"
,
"Cython.Compiler.ParseTreeTransforms"
,
"Cython.Compiler.ParseTreeTransforms"
,
"Cython.Compiler.Nodes"
,
"Cython.Compiler.Nodes"
,
"Cython.Compiler.ExprNodes"
,
"Cython.Compiler.ExprNodes"
,
...
...
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