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
2a6bfe21
Commit
2a6bfe21
authored
Nov 26, 2009
by
Stefan Behnel
Browse files
Options
Browse Files
Download
Plain Diff
merge
parents
017241bf
4fa6d965
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
158 additions
and
34 deletions
+158
-34
Cython/Compiler/Optimize.py
Cython/Compiler/Optimize.py
+70
-31
Cython/Compiler/Symtab.py
Cython/Compiler/Symtab.py
+5
-3
tests/run/builtinnames.pyx
tests/run/builtinnames.pyx
+18
-0
tests/run/charptr_len.pyx
tests/run/charptr_len.pyx
+65
-0
No files found.
Cython/Compiler/Optimize.py
View file @
2a6bfe21
...
...
@@ -740,19 +740,22 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
arg_tuple
=
node
.
positional_args
if
not
isinstance
(
arg_tuple
,
ExprNodes
.
TupleNode
):
return
node
args
=
arg_tuple
.
args
return
self
.
_dispatch_to_handler
(
node
,
function
,
arg
_tuple
,
node
.
keyword_args
)
node
,
function
,
arg
s
,
node
.
keyword_args
)
def
visit_SimpleCallNode
(
self
,
node
):
self
.
visitchildren
(
node
)
function
=
node
.
function
if
not
function
.
type
.
is_pyobject
:
return
node
arg_tuple
=
node
.
arg_tuple
if
not
isinstance
(
arg_tuple
,
ExprNodes
.
TupleNode
):
return
node
if
function
.
type
.
is_pyobject
:
arg_tuple
=
node
.
arg_tuple
if
not
isinstance
(
arg_tuple
,
ExprNodes
.
TupleNode
):
return
node
args
=
arg_tuple
.
args
else
:
args
=
node
.
args
return
self
.
_dispatch_to_handler
(
node
,
node
.
function
,
arg_tuple
)
node
,
function
,
args
)
### cleanup to avoid redundant coercions to/from Python types
...
...
@@ -823,20 +826,25 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
handler
=
getattr
(
self
,
'_handle_any_%s'
%
match_name
,
None
)
return
handler
def
_dispatch_to_handler
(
self
,
node
,
function
,
arg_
tuple
,
kwargs
=
None
):
def
_dispatch_to_handler
(
self
,
node
,
function
,
arg_
list
,
kwargs
=
None
):
if
function
.
is_name
:
match_name
=
"_function_%s"
%
function
.
name
# we only consider functions that are either builtin
# Python functions or builtins that were already replaced
# into a C function call (defined in the builtin scope)
is_builtin
=
function
.
entry
.
is_builtin
\
or
getattr
(
function
.
entry
,
'scope'
,
None
)
is
Builtin
.
builtin_scope
if
not
is_builtin
:
return
node
function_handler
=
self
.
_find_handler
(
"function_%s"
%
function
.
name
,
kwargs
)
if
function_handler
is
None
:
return
node
if
kwargs
:
return
function_handler
(
node
,
arg_
tuple
,
kwargs
)
return
function_handler
(
node
,
arg_
list
,
kwargs
)
else
:
return
function_handler
(
node
,
arg_
tuple
)
elif
function
.
is_attribute
:
return
function_handler
(
node
,
arg_
list
)
elif
function
.
is_attribute
and
function
.
type
.
is_pyobject
:
attr_name
=
function
.
attribute
arg_list
=
arg_tuple
.
args
self_arg
=
function
.
obj
obj_type
=
self_arg
.
type
is_unbound_method
=
False
...
...
@@ -892,7 +900,7 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
"""Replace dict(a=b,c=d,...) by the underlying keyword dict
construction which is done anyway.
"""
if
len
(
pos_args
.
args
)
>
0
:
if
len
(
pos_args
)
>
0
:
return
node
if
not
isinstance
(
kwargs
,
ExprNodes
.
DictNode
):
return
node
...
...
@@ -910,9 +918,9 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
"""Replace dict(some_dict) by PyDict_Copy(some_dict) and
dict([ (a,b) for ... ]) by a literal { a:b for ... }.
"""
if
len
(
pos_args
.
args
)
!=
1
:
if
len
(
pos_args
)
!=
1
:
return
node
arg
=
pos_args
.
args
[
0
]
arg
=
pos_args
[
0
]
if
arg
.
type
is
Builtin
.
dict_type
:
arg
=
ExprNodes
.
NoneCheckNode
(
arg
,
"PyExc_TypeError"
,
"'NoneType' is not iterable"
)
...
...
@@ -943,13 +951,13 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
"""Replace set([a,b,...]) by a literal set {a,b,...} and
set([ x for ... ]) by a literal { x for ... }.
"""
arg_count
=
len
(
pos_args
.
args
)
arg_count
=
len
(
pos_args
)
if
arg_count
==
0
:
return
ExprNodes
.
SetNode
(
node
.
pos
,
args
=
[],
type
=
Builtin
.
set_type
,
is_temp
=
1
)
if
arg_count
>
1
:
return
node
iterable
=
pos_args
.
args
[
0
]
iterable
=
pos_args
[
0
]
if
isinstance
(
iterable
,
(
ExprNodes
.
ListNode
,
ExprNodes
.
TupleNode
)):
return
ExprNodes
.
SetNode
(
node
.
pos
,
args
=
iterable
.
args
,
type
=
Builtin
.
set_type
,
is_temp
=
1
)
...
...
@@ -971,20 +979,20 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
def
_handle_simple_function_tuple
(
self
,
node
,
pos_args
):
"""Replace tuple([...]) by a call to PyList_AsTuple.
"""
if
len
(
pos_args
.
args
)
!=
1
:
if
len
(
pos_args
)
!=
1
:
return
node
list_arg
=
pos_args
.
args
[
0
]
list_arg
=
pos_args
[
0
]
if
list_arg
.
type
is
not
Builtin
.
list_type
:
return
node
if
not
isinstance
(
list_arg
,
(
ExprNodes
.
ComprehensionNode
,
ExprNodes
.
ListNode
)):
pos_args
.
args
[
0
]
=
ExprNodes
.
NoneCheckNode
(
pos_args
[
0
]
=
ExprNodes
.
NoneCheckNode
(
list_arg
,
"PyExc_TypeError"
,
"'NoneType' object is not iterable"
)
return
ExprNodes
.
PythonCapiCallNode
(
node
.
pos
,
"PyList_AsTuple"
,
self
.
PyList_AsTuple_func_type
,
args
=
pos_args
.
args
,
args
=
pos_args
,
is_temp
=
node
.
is_temp
)
...
...
@@ -1004,22 +1012,21 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
])
def
_handle_simple_function_getattr
(
self
,
node
,
pos_args
):
args
=
pos_args
.
args
if
len
(
args
)
==
2
:
if
len
(
pos_args
)
==
2
:
node
=
ExprNodes
.
PythonCapiCallNode
(
node
.
pos
,
"PyObject_GetAttr"
,
self
.
PyObject_GetAttr2_func_type
,
args
=
args
,
args
=
pos_
args
,
is_temp
=
node
.
is_temp
)
elif
len
(
args
)
==
3
:
elif
len
(
pos_
args
)
==
3
:
node
=
ExprNodes
.
PythonCapiCallNode
(
node
.
pos
,
"__Pyx_GetAttr3"
,
self
.
PyObject_GetAttr3_func_type
,
utility_code
=
Builtin
.
getattr3_utility_code
,
args
=
args
,
args
=
pos_
args
,
is_temp
=
node
.
is_temp
)
else
:
self
.
_error_wrong_arg_count
(
'getattr'
,
node
,
args
,
'2 or 3'
)
self
.
_error_wrong_arg_count
(
'getattr'
,
node
,
pos_
args
,
'2 or 3'
)
return
node
Pyx_Type_func_type
=
PyrexTypes
.
CFuncType
(
...
...
@@ -1028,17 +1035,42 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
])
def
_handle_simple_function_type
(
self
,
node
,
pos_args
):
args
=
pos_args
.
args
if
len
(
args
)
!=
1
:
if
len
(
pos_args
)
!=
1
:
return
node
node
=
ExprNodes
.
PythonCapiCallNode
(
node
.
pos
,
"__Pyx_Type"
,
self
.
Pyx_Type_func_type
,
args
=
args
,
args
=
pos_
args
,
is_temp
=
node
.
is_temp
,
utility_code
=
pytype_utility_code
,
)
return
node
Pyx_strlen_func_type
=
PyrexTypes
.
CFuncType
(
PyrexTypes
.
c_size_t_type
,
[
PyrexTypes
.
CFuncTypeArg
(
"bytes"
,
PyrexTypes
.
c_char_ptr_type
,
None
)
])
def
_handle_simple_function_len
(
self
,
node
,
pos_args
):
# note: this only works because we already replaced len() by
# PyObject_Length() which returns a Py_ssize_t instead of a
# Python object, so we can return a plain size_t instead
# without caring about Python object conversion etc.
if
len
(
pos_args
)
!=
1
:
self
.
_error_wrong_arg_count
(
'len'
,
node
,
pos_args
,
1
)
return
node
arg
=
pos_args
[
0
]
if
isinstance
(
arg
,
ExprNodes
.
CoerceToPyTypeNode
):
arg
=
arg
.
arg
if
not
arg
.
type
.
is_string
:
return
node
node
=
ExprNodes
.
PythonCapiCallNode
(
node
.
pos
,
"strlen"
,
self
.
Pyx_strlen_func_type
,
args
=
[
arg
],
is_temp
=
node
.
is_temp
,
utility_code
=
include_string_h_utility_code
,
)
return
node
### special methods
Pyx_tp_new_func_type
=
PyrexTypes
.
CFuncType
(
...
...
@@ -1477,6 +1509,13 @@ static INLINE PyObject* __Pyx_Type(PyObject* o) {
)
include_string_h_utility_code
=
UtilityCode
(
proto
=
"""
#include <string.h>
"""
)
tpnew_utility_code
=
UtilityCode
(
proto
=
"""
static INLINE PyObject* __Pyx_tp_new(PyObject* type_obj) {
...
...
Cython/Compiler/Symtab.py
View file @
2a6bfe21
...
...
@@ -624,6 +624,7 @@ class BuiltinScope(Scope):
var_entry.is_variable = 1
var_entry.is_cglobal = 1
var_entry.is_readonly = 1
var_entry.is_builtin = 1
var_entry.utility_code = utility_code
entry.as_variable = var_entry
...
...
@@ -1184,6 +1185,9 @@ class ClassScope(Scope):
return self.outer_scope.add_string_const(value, identifier)
def lookup(self, name):
entry = Scope.lookup(self, name)
if entry:
return entry
if name == "classmethod":
# We don'
t
want
to
use
the
builtin
classmethod
here
'cause it won'
t
do
the
# right thing in this scope (as the class memebers aren't still functions).
...
...
@@ -1197,9 +1201,7 @@ class ClassScope(Scope):
py_object_type
,
[
PyrexTypes
.
CFuncTypeArg
(
""
,
py_object_type
,
None
)],
0
,
0
))
entry
.
is_cfunction
=
1
return
entry
else
:
return
Scope
.
lookup
(
self
,
name
)
return
entry
class
PyClassScope
(
ClassScope
):
...
...
tests/run/builtinnames.pyx
View file @
2a6bfe21
cimport
cython
def
test_file_py
(
file
):
assert
isinstance
(
file
,
(
str
,
unicode
)),
\
u"not a string, found '%s' instead"
%
file
.
__class__
.
__name__
...
...
@@ -12,19 +14,35 @@ cdef test_file_c(file):
def
range
(
arg
):
return
u'range'
+
arg
def
len
(
arg
):
return
u'len'
+
arg
cdef
type
(
arg
):
return
u'type'
+
arg
@
cython
.
test_fail_if_path_exists
(
'//SimpleCallNode/NameNode[@name="type" and @entry.is_cfunction]'
,
'//SimpleCallNode/NameNode[@name="len" and @entry.is_cfunction]'
,
)
@
cython
.
test_assert_path_exists
(
'//SimpleCallNode/NameNode[@name="type"]'
,
'//SimpleCallNode/NameNode[@name="type" and not(@entry.is_cfunction)]'
,
'//SimpleCallNode/NameNode[@name="len"]'
,
)
def
test_c
(
arg
):
"""
>>> test_c('abc')
fileabc
lenabc
typeabc
>>> print(test_file_py('abc'))
abc
>>> print(range('abc'))
rangeabc
>>> print(len('abc'))
lenabc
"""
print
test_file_c
(
arg
)
print
len
(
arg
)
print
type
(
arg
)
tests/run/charptr_len.pyx
0 → 100644
View file @
2a6bfe21
__doc__
=
"""
>>> lentest_char()
7
>>> lentest_char_c()
7
>>> lentest_uchar()
7
>>> lentest_uchar_c()
7
>>> lentest_py()
7
>>> lentest_py_c()
7
"""
cimport
cython
cdef
char
*
s
=
b"abcdefg"
cdef
unsigned
char
*
us
=
b"abcdefg"
cdef
bytes
pystr
=
b"abcdefg"
@
cython
.
test_assert_path_exists
(
"//PythonCapiCallNode"
,
)
def
lentest_char
():
return
len
(
s
)
@
cython
.
test_assert_path_exists
(
"//PythonCapiCallNode"
,
)
def
lentest_char_c
():
cdef
Py_ssize_t
l
=
len
(
s
)
return
l
@
cython
.
test_assert_path_exists
(
"//PythonCapiCallNode"
,
)
def
lentest_uchar
():
return
len
(
us
)
@
cython
.
test_assert_path_exists
(
"//PythonCapiCallNode"
,
)
def
lentest_uchar_c
():
cdef
Py_ssize_t
l
=
len
(
us
)
return
l
@
cython
.
test_assert_path_exists
(
"//SimpleCallNode"
,
)
def
lentest_py
():
return
len
(
pystr
)
@
cython
.
test_assert_path_exists
(
"//SimpleCallNode"
,
)
def
lentest_py_c
():
cdef
Py_ssize_t
l
=
len
(
pystr
)
return
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