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
b978c7e0
Commit
b978c7e0
authored
Feb 19, 2010
by
Robert Bradshaw
Browse files
Options
Browse Files
Download
Plain Diff
merge
parents
0a0e72a7
fa55b747
Changes
25
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
25 changed files
with
1633 additions
and
156 deletions
+1633
-156
Cython/Compiler/AutoDocTransforms.py
Cython/Compiler/AutoDocTransforms.py
+3
-3
Cython/Compiler/Code.py
Cython/Compiler/Code.py
+3
-1
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+47
-3
Cython/Compiler/Main.py
Cython/Compiler/Main.py
+5
-1
Cython/Compiler/Nodes.py
Cython/Compiler/Nodes.py
+18
-6
Cython/Compiler/Optimize.py
Cython/Compiler/Optimize.py
+36
-19
Cython/Compiler/Parsing.py
Cython/Compiler/Parsing.py
+9
-0
Cython/Compiler/PyrexTypes.py
Cython/Compiler/PyrexTypes.py
+13
-0
Cython/Compiler/Scanning.py
Cython/Compiler/Scanning.py
+4
-1
Cython/Compiler/StringEncoding.py
Cython/Compiler/StringEncoding.py
+2
-2
Cython/Compiler/Symtab.py
Cython/Compiler/Symtab.py
+5
-1
Cython/Compiler/TypeInference.py
Cython/Compiler/TypeInference.py
+77
-5
Cython/Distutils/__init__.py
Cython/Distutils/__init__.py
+2
-1
Cython/Distutils/build_ext.py
Cython/Distutils/build_ext.py
+10
-16
Tools/cython-mode.el
Tools/cython-mode.el
+63
-1
runtests.py
runtests.py
+2
-2
setup.py
setup.py
+76
-41
tests/compile/declarations.pyx
tests/compile/declarations.pyx
+4
-0
tests/compile/msvc_strings.pyx
tests/compile/msvc_strings.pyx
+1072
-0
tests/run/carray_slicing.pyx
tests/run/carray_slicing.pyx
+80
-17
tests/run/complex_numbers_T305.pyx
tests/run/complex_numbers_T305.pyx
+1
-1
tests/run/cpp_nested_templates.pyx
tests/run/cpp_nested_templates.pyx
+2
-5
tests/run/cpp_templates.pyx
tests/run/cpp_templates.pyx
+0
-5
tests/run/extern_builtins_T258.pyx
tests/run/extern_builtins_T258.pyx
+4
-18
tests/run/type_inference.pyx
tests/run/type_inference.pyx
+95
-7
No files found.
Cython/Compiler/AutoDocTransforms.py
View file @
b978c7e0
...
...
@@ -20,7 +20,7 @@ class EmbedSignature(CythonTransform):
try
:
denv
=
self
.
denv
# XXX
ctval
=
default_val
.
compile_time_value
(
self
.
denv
)
repr_val
=
'%r'
%
ctval
repr_val
=
repr
(
ctval
)
if
isinstance
(
default_val
,
ExprNodes
.
UnicodeNode
):
if
repr_val
[:
1
]
!=
'u'
:
return
u'u%s'
%
repr_val
...
...
@@ -28,8 +28,8 @@ class EmbedSignature(CythonTransform):
if
repr_val
[:
1
]
!=
'b'
:
return
u'b%s'
%
repr_val
elif
isinstance
(
default_val
,
ExprNodes
.
StringNode
):
if
repr_val
[:
1
]
in
(
'u'
,
'b'
)
:
repr_val
[
1
:]
if
repr_val
[:
1
]
in
'ub'
:
re
turn
re
pr_val
[
1
:]
return
repr_val
except
Exception
:
try
:
...
...
Cython/Compiler/Code.py
View file @
b978c7e0
...
...
@@ -637,6 +637,8 @@ class GlobalState(object):
def put_cached_builtin_init(self, pos, name, cname):
w = self.parts['cached_builtins']
interned_cname = self.get_interned_identifier(name).cname
from ExprNodes import get_name_interned_utility_code
self.use_utility_code(get_name_interned_utility_code)
w.putln('%s = __Pyx_GetName(%s, %s); if (!%s) %s' % (
cname,
Naming.builtins_cname,
...
...
@@ -667,7 +669,7 @@ class GlobalState(object):
decls_writer = self.parts['decls']
for _, cname, c in c_consts:
decls_writer.putln('static char %s[] = "
%
s
";' % (
cname, StringEncoding.split_
docstring
(c.escaped_value)))
cname, StringEncoding.split_
string_literal
(c.escaped_value)))
if c.py_strings is not None:
for py_string in c.py_strings.itervalues():
py_strings.append((c.cname, len(py_string.cname), py_string))
...
...
Cython/Compiler/ExprNodes.py
View file @
b978c7e0
...
...
@@ -1246,8 +1246,8 @@ class NameNode(AtomicExprNode):
self
.
is_temp
=
0
else
:
self
.
is_temp
=
1
self
.
is_used_as_rvalue
=
1
env
.
use_utility_code
(
get_name_interned_utility_code
)
self
.
is_used_as_rvalue
=
1
def
nogil_check
(
self
,
env
):
if
self
.
is_used_as_rvalue
:
...
...
@@ -1334,6 +1334,7 @@ class NameNode(AtomicExprNode):
namespace
=
Naming
.
builtins_cname
else
:
# entry.is_pyglobal
namespace
=
entry
.
scope
.
namespace_cname
code
.
globalstate
.
use_utility_code
(
get_name_interned_utility_code
)
code
.
putln
(
'%s = __Pyx_GetName(%s, %s); %s'
%
(
self
.
result
(),
...
...
@@ -1554,6 +1555,12 @@ class IteratorNode(ExprNode):
def
analyse_types
(
self
,
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
.
is_temp
=
1
...
...
@@ -2444,6 +2451,15 @@ class CallNode(ExprNode):
self
.
analyse_types
(
env
)
self
.
coerce_to
(
type
,
env
)
return
True
elif
type
and
type
.
is_cpp_class
:
for
arg
in
self
.
args
:
arg
.
analyse_types
(
env
)
constructor
=
type
.
scope
.
lookup
(
"<init>"
)
self
.
function
=
RawCNameExprNode
(
self
.
function
.
pos
,
constructor
.
type
)
self
.
function
.
entry
=
constructor
self
.
function
.
set_cname
(
type
.
declaration_code
(
""
))
self
.
analyse_c_function_call
(
env
)
return
True
def
nogil_check
(
self
,
env
):
func_type
=
self
.
function_type
()
...
...
@@ -2490,6 +2506,8 @@ class SimpleCallNode(CallNode):
def
infer_type
(
self
,
env
):
function
=
self
.
function
func_type
=
function
.
infer_type
(
env
)
if
isinstance
(
self
.
function
,
NewExprNode
):
return
PyrexTypes
.
CPtrType
(
self
.
function
.
class_type
)
if
func_type
.
is_ptr
:
func_type
=
func_type
.
base_type
if
func_type
.
is_cfunction
:
...
...
@@ -4708,7 +4726,9 @@ class BinopNode(ExprNode):
self
.
operand2
.
analyse_types
(
env
)
if
self
.
is_py_operation
():
self
.
coerce_operands_to_pyobjects
(
env
)
self
.
type
=
py_object_type
self
.
type
=
self
.
result_type
(
self
.
operand1
.
type
,
self
.
operand2
.
type
)
assert
self
.
type
.
is_pyobject
self
.
is_temp
=
1
elif
self
.
is_cpp_operation
():
self
.
analyse_cpp_operation
(
env
)
...
...
@@ -4750,6 +4770,30 @@ class BinopNode(ExprNode):
def
result_type
(
self
,
type1
,
type2
):
if
self
.
is_py_operation_types
(
type1
,
type2
):
if
type2
.
is_string
:
type2
=
Builtin
.
bytes_type
if
type1
.
is_string
:
type1
=
Builtin
.
bytes_type
elif
self
.
operator
==
'%'
\
and
type1
in
(
Builtin
.
str_type
,
Builtin
.
unicode_type
):
# note that b'%s' % b'abc' doesn't work in Py3
return
type1
if
type1
.
is_builtin_type
:
if
type1
is
type2
:
if
self
.
operator
in
'**%+|&^'
:
# FIXME: at least these operators should be safe - others?
return
type1
elif
self
.
operator
==
'*'
:
if
type1
in
(
Builtin
.
bytes_type
,
Builtin
.
str_type
,
Builtin
.
unicode_type
):
return
type1
# multiplication of containers/numbers with an
# integer value always (?) returns the same type
if
type2
.
is_int
:
return
type1
elif
type2
.
is_builtin_type
and
type1
.
is_int
and
self
.
operator
==
'*'
:
# multiplication of containers/numbers with an
# integer value always (?) returns the same type
return
type2
return
py_object_type
else
:
return
self
.
compute_c_result_type
(
type1
,
type2
)
...
...
Cython/Compiler/Main.py
View file @
b978c7e0
...
...
@@ -88,7 +88,7 @@ class Context(object):
from
ParseTreeTransforms
import
AnalyseDeclarationsTransform
,
AnalyseExpressionsTransform
from
ParseTreeTransforms
import
CreateClosureClasses
,
MarkClosureVisitor
,
DecoratorTransform
from
ParseTreeTransforms
import
InterpretCompilerDirectives
,
TransformBuiltinMethods
from
TypeInference
import
MarkAssignments
from
TypeInference
import
MarkAssignments
,
MarkOverflowingArithmatic
from
ParseTreeTransforms
import
AlignFunctionDefinitions
,
GilCheck
from
AnalysedTreeTransforms
import
AutoTestDictTransform
from
AutoDocTransforms
import
EmbedSignature
...
...
@@ -135,6 +135,7 @@ class Context(object):
EmbedSignature
(
self
),
EarlyReplaceBuiltinCalls
(
self
),
MarkAssignments
(
self
),
MarkOverflowingArithmatic
(
self
),
TransformBuiltinMethods
(
self
),
IntroduceBufferAuxiliaryVars
(
self
),
_check_c_declarations
,
...
...
@@ -218,8 +219,11 @@ class Context(object):
for
phase
in
pipeline
:
if
phase
is
not
None
:
if
DebugFlags
.
debug_verbose_pipeline
:
t
=
time
()
print
"Entering pipeline phase %r"
%
phase
data
=
phase
(
data
)
if
DebugFlags
.
debug_verbose_pipeline
:
print
" %.3f seconds"
%
(
time
()
-
t
)
except
CompileError
,
err
:
# err is set
Errors
.
report_error
(
err
)
...
...
Cython/Compiler/Nodes.py
View file @
b978c7e0
...
...
@@ -22,7 +22,7 @@ from Symtab import ModuleScope, LocalScope, GeneratorLocalScope, \
StructOrUnionScope
,
PyClassScope
,
CClassScope
,
CppClassScope
from
Cython.Utils
import
open_new_file
,
replace_suffix
from
Code
import
UtilityCode
from
StringEncoding
import
EncodedString
,
escape_byte_string
,
split_
docstring
from
StringEncoding
import
EncodedString
,
escape_byte_string
,
split_
string_literal
import
Options
import
ControlFlow
import
DebugFlags
...
...
@@ -661,6 +661,17 @@ class CArgDeclNode(Node):
base_type
=
self
.
base_type
.
analyse
(
env
,
could_be_name
=
could_be_name
)
if
hasattr
(
self
.
base_type
,
'arg_name'
)
and
self
.
base_type
.
arg_name
:
self
.
declarator
.
name
=
self
.
base_type
.
arg_name
# The parser is unable to resolve the ambiguity of [] as part of the
# type (e.g. in buffers) or empty declarator (as with arrays).
# This is only arises for empty multi-dimensional arrays.
if
(
base_type
.
is_array
and
isinstance
(
self
.
base_type
,
TemplatedTypeNode
)
and
isinstance
(
self
.
declarator
,
CArrayDeclaratorNode
)):
declarator
=
self
.
declarator
while
isinstance
(
declarator
.
base
,
CArrayDeclaratorNode
):
declarator
=
declarator
.
base
declarator
.
base
=
self
.
base_type
.
array_declarator
base_type
=
base_type
.
base_type
return
self
.
declarator
.
analyse
(
base_type
,
env
,
nonempty
=
nonempty
)
else
:
return
self
.
name_declarator
,
self
.
type
...
...
@@ -821,7 +832,7 @@ class TemplatedTypeNode(CBaseTypeNode):
else
:
# Array
empty_declarator
=
CNameDeclaratorNode
(
self
.
pos
,
name
=
""
)
empty_declarator
=
CNameDeclaratorNode
(
self
.
pos
,
name
=
""
,
cname
=
None
)
if
len
(
self
.
positional_args
)
>
1
or
self
.
keyword_args
.
key_value_pairs
:
error
(
self
.
pos
,
"invalid array declaration"
)
self
.
type
=
PyrexTypes
.
error_type
...
...
@@ -832,9 +843,10 @@ class TemplatedTypeNode(CBaseTypeNode):
dimension
=
None
else
:
dimension
=
self
.
positional_args
[
0
]
self
.
type
=
CArrayDeclaratorNode
(
self
.
pos
,
self
.
array_declarator
=
CArrayDeclaratorNode
(
self
.
pos
,
base
=
empty_declarator
,
dimension
=
dimension
).
analyse
(
base_type
,
env
)[
1
]
dimension
=
dimension
)
self
.
type
=
self
.
array_declarator
.
analyse
(
base_type
,
env
)[
1
]
return
self
.
type
...
...
@@ -2052,7 +2064,7 @@ class DefNode(FuncDefNode):
code
.
putln
(
'static char %s[] = "%s";'
%
(
self
.
entry
.
doc_cname
,
split_
docstring
(
escape_byte_string
(
docstr
))))
split_
string_literal
(
escape_byte_string
(
docstr
))))
if
with_pymethdef
:
code
.
put
(
"static PyMethodDef %s = "
%
...
...
@@ -4166,7 +4178,7 @@ class ForFromStatNode(LoopNode, StatNode):
target_node
=
ExprNodes
.
PyTempNode
(
self
.
target
.
pos
,
None
)
target_node
.
allocate
(
code
)
interned_cname
=
code
.
intern_identifier
(
self
.
target
.
entry
.
name
)
code
.
putln
(
"/*here*/"
)
code
.
globalstate
.
use_utility_code
(
ExprNodes
.
get_name_interned_utility_code
)
code
.
putln
(
"%s = __Pyx_GetName(%s, %s); %s"
%
(
target_node
.
result
(),
Naming
.
module_cname
,
...
...
Cython/Compiler/Optimize.py
View file @
b978c7e0
...
...
@@ -89,10 +89,12 @@ class IterationTransform(Visitor.VisitorTransform):
return
self
.
_transform_dict_iteration
(
node
,
dict_obj
=
iterator
,
keys
=
True
,
values
=
False
)
# C array
slice
iteration?
# C array
(slice)
iteration?
if
isinstance
(
iterator
,
ExprNodes
.
SliceIndexNode
)
and
\
(
iterator
.
base
.
type
.
is_array
or
iterator
.
base
.
type
.
is_ptr
):
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
):
return
node
...
...
@@ -131,13 +133,26 @@ class IterationTransform(Visitor.VisitorTransform):
return
node
def
_transform_carray_iteration
(
self
,
node
,
slice_node
):
if
isinstance
(
slice_node
,
ExprNodes
.
SliceIndexNode
):
slice_base
=
slice_node
.
base
start
=
slice_node
.
start
stop
=
slice_node
.
stop
step
=
None
if
not
stop
:
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
:
start_ptr_node
=
ExprNodes
.
AddNode
(
...
...
@@ -145,7 +160,7 @@ class IterationTransform(Visitor.VisitorTransform):
operand1
=
carray_ptr
,
operator
=
'+'
,
operand2
=
start
,
type
=
carray_ptr
.
type
)
type
=
ptr_
type
)
else
:
start_ptr_node
=
carray_ptr
...
...
@@ -154,13 +169,13 @@ class IterationTransform(Visitor.VisitorTransform):
operand1
=
carray_ptr
,
operator
=
'+'
,
operand2
=
stop
,
type
=
carray_ptr
.
type
type
=
ptr_
type
).
coerce_to_simple
(
self
.
current_scope
)
counter
=
UtilNodes
.
TempHandle
(
carray_ptr
.
type
)
counter
=
UtilNodes
.
TempHandle
(
ptr_
type
)
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
target_value
=
ExprNodes
.
SliceIndexNode
(
node
.
target
.
pos
,
...
...
@@ -181,7 +196,7 @@ class IterationTransform(Visitor.VisitorTransform):
type
=
PyrexTypes
.
c_int_type
),
base
=
counter_temp
,
is_buffer_access
=
False
,
type
=
carray_ptr
.
type
.
base_type
)
type
=
ptr_
type
.
base_type
)
if
target_value
.
type
!=
node
.
target
.
type
:
target_value
=
target_value
.
coerce_to
(
node
.
target
.
type
,
...
...
@@ -1606,20 +1621,20 @@ impl = ""
pop_utility_code
=
UtilityCode
(
proto
=
"""
static CYTHON_INLINE PyObject* __Pyx_PyObject_Pop(PyObject* L) {
#if PY_VERSION_HEX >= 0x02040000
if (likely(PyList_CheckExact(L))
/* Check that both the size is positive and no reallocation shrinking needs to be done. */
&& likely(PyList_GET_SIZE(L) > (((PyListObject*)L)->allocated >> 1))) {
Py_SIZE(L) -= 1;
return PyList_GET_ITEM(L, PyList_GET_SIZE(L));
}
else {
#endif
PyObject *r, *m;
m = __Pyx_GetAttrString(L, "pop");
if (!m) return NULL;
r = PyObject_CallObject(m, NULL);
Py_DECREF(m);
return r;
}
}
"""
,
impl
=
""
...
...
@@ -1632,6 +1647,7 @@ static PyObject* __Pyx_PyObject_PopIndex(PyObject* L, Py_ssize_t ix);
impl
=
"""
static PyObject* __Pyx_PyObject_PopIndex(PyObject* L, Py_ssize_t ix) {
PyObject *r, *m, *t, *py_ix;
#if PY_VERSION_HEX >= 0x02040000
if (likely(PyList_CheckExact(L))) {
Py_ssize_t size = PyList_GET_SIZE(L);
if (likely(size > (((PyListObject*)L)->allocated >> 1))) {
...
...
@@ -1650,6 +1666,7 @@ static PyObject* __Pyx_PyObject_PopIndex(PyObject* L, Py_ssize_t ix) {
}
}
}
#endif
py_ix = t = NULL;
m = __Pyx_GetAttrString(L, "pop");
if (!m) goto bad;
...
...
Cython/Compiler/Parsing.py
View file @
b978c7e0
...
...
@@ -10,6 +10,15 @@ cython.declare(Nodes=object, ExprNodes=object, EncodedString=object)
import
os
import
re
import
sys
try
:
from
__builtin__
import
set
except
ImportError
:
try
:
from
builtins
import
set
except
ImportError
:
from
sets
import
Set
as
set
from
Cython.Compiler.Scanning
import
PyrexScanner
,
FileSourceDescriptor
import
Nodes
import
ExprNodes
...
...
Cython/Compiler/PyrexTypes.py
View file @
b978c7e0
...
...
@@ -1893,6 +1893,8 @@ class CppClassType(CType):
def
assignable_from_resolved_type
(
self
,
other_type
):
# TODO: handle operator=(...) here?
if
other_type
is
error_type
:
return
True
return
other_type
.
is_cpp_class
and
other_type
.
is_subclass
(
self
)
def
attributes_known
(
self
):
...
...
@@ -2369,6 +2371,17 @@ def spanning_type(type1, type2):
return
widest_numeric_type
(
type1
,
c_double_type
)
elif
type1
.
is_pyobject
^
type2
.
is_pyobject
:
return
py_object_type
elif
type1
.
is_extension_type
and
type2
.
is_extension_type
:
if
type1
.
typeobj_is_imported
()
or
type2
.
typeobj_is_imported
():
return
py_object_type
while
True
:
if
type1
.
subtype_of
(
type2
):
return
type2
elif
type2
.
subtype_of
(
type1
):
return
type1
type1
,
type2
=
type1
.
base_type
,
type2
.
base_type
if
type1
is
None
or
type2
is
None
:
return
py_object_type
elif
type1
.
assignable_from
(
type2
):
if
type1
.
is_extension_type
and
type1
.
typeobj_is_imported
():
# external types are unsafe, so we use PyObject instead
...
...
Cython/Compiler/Scanning.py
View file @
b978c7e0
...
...
@@ -97,7 +97,10 @@ def initial_compile_time_env():
'UNAME_VERSION'
,
'UNAME_MACHINE'
)
for
name
,
value
in
zip
(
names
,
platform
.
uname
()):
benv
.
declare
(
name
,
value
)
try
:
import
__builtin__
as
builtins
except
ImportError
:
import
builtins
names
=
(
'False'
,
'True'
,
'abs'
,
'bool'
,
'chr'
,
'cmp'
,
'complex'
,
'dict'
,
'divmod'
,
'enumerate'
,
'float'
,
'hash'
,
'hex'
,
'int'
,
'len'
,
'list'
,
'long'
,
'map'
,
'max'
,
'min'
,
...
...
Cython/Compiler/StringEncoding.py
View file @
b978c7e0
...
...
@@ -185,9 +185,9 @@ def escape_byte_string(s):
append
(
c
)
return
join_bytes
(
l
).
decode
(
'ISO-8859-1'
)
def
split_
docstring
(
s
):
def
split_
string_literal
(
s
):
# MSVC can't handle long string literals.
if
len
(
s
)
<
2047
:
return
s
else
:
return
'""'
.
join
([
s
[
i
:
i
+
2000
]
for
i
in
range
(
0
,
len
(
s
),
2000
)])
return
'""'
.
join
([
s
[
i
:
i
+
2000
]
for
i
in
range
(
0
,
len
(
s
),
2000
)])
.
replace
(
r'\""'
,
'""
\
\
'
)
Cython/Compiler/Symtab.py
View file @
b978c7e0
...
...
@@ -119,6 +119,8 @@ class Entry(object):
# inline_func_in_pxd boolean Hacky special case for inline function in pxd file.
# Ideally this should not be necesarry.
# assignments [ExprNode] List of expressions that get assigned to this entry.
# might_overflow boolean In an arithmatic expression that could cause
# overflow (used for type inference).
inline_func_in_pxd = False
borrowed = 0
...
...
@@ -167,6 +169,7 @@ class Entry(object):
is_overridable = 0
buffer_aux = None
prev_entry = None
might_overflow = 0
def __init__(self, name, cname, type, pos = None, init = None):
self.name = name
...
...
@@ -434,7 +437,7 @@ class Scope(object):
if type.is_cpp_class and visibility != 'extern':
constructor = type.scope.lookup(u'<init>')
if constructor is not None and PyrexTypes.best_match([], constructor.all_alternatives()) is None:
error(pos, "
C
++
class
must
have
a
n
empty
constructor
to
be
stack
allocated
")
error(pos, "
C
++
class
must
have
a
default
constructor
to
be
stack
allocated
")
entry = self.declare(name, cname, type, pos, visibility)
entry.is_variable = 1
self.control_flow.set_state((), (name, 'initalized'), False)
...
...
@@ -1564,6 +1567,7 @@ class CppClassScope(Scope):
if
name
==
self
.
name
.
split
(
'::'
)[
-
1
]
and
cname
is
None
:
self
.
check_base_default_constructor
(
pos
)
name
=
'<init>'
type
.
return_type
=
self
.
lookup
(
self
.
name
).
type
prev_entry
=
self
.
lookup_here
(
name
)
entry
=
self
.
declare_var
(
name
,
type
,
pos
,
cname
,
visibility
)
if
prev_entry
:
...
...
Cython/Compiler/TypeInference.py
View file @
b978c7e0
...
...
@@ -112,6 +112,62 @@ class MarkAssignments(CythonTransform):
self
.
visitchildren
(
node
)
return
node
class
MarkOverflowingArithmatic
(
CythonTransform
):
# It may be possible to integrate this with the above for
# performance improvements (though likely not worth it).
might_overflow
=
False
def
__call__
(
self
,
root
):
self
.
env_stack
=
[]
self
.
env
=
root
.
scope
return
super
(
MarkOverflowingArithmatic
,
self
).
__call__
(
root
)
def
visit_safe_node
(
self
,
node
):
self
.
might_overflow
,
saved
=
False
,
self
.
might_overflow
self
.
visitchildren
(
node
)
self
.
might_overflow
=
saved
return
node
def
visit_neutral_node
(
self
,
node
):
self
.
visitchildren
(
node
)
return
node
def
visit_dangerous_node
(
self
,
node
):
self
.
might_overflow
,
saved
=
True
,
self
.
might_overflow
self
.
visitchildren
(
node
)
self
.
might_overflow
=
saved
return
node
def
visit_FuncDefNode
(
self
,
node
):
self
.
env_stack
.
append
(
self
.
env
)
self
.
env
=
node
.
local_scope
self
.
visit_safe_node
(
node
)
self
.
env
=
self
.
env_stack
.
pop
()
return
node
def
visit_NameNode
(
self
,
node
):
if
self
.
might_overflow
:
entry
=
node
.
entry
or
self
.
env
.
lookup
(
node
.
name
)
if
entry
:
entry
.
might_overflow
=
True
return
node
def
visit_BinopNode
(
self
,
node
):
if
node
.
operator
in
'&|^'
:
return
self
.
visit_neutral_node
(
node
)
else
:
return
self
.
visit_dangerous_node
(
node
)
visit_UnopNode
=
visit_neutral_node
visit_UnaryMinusNode
=
visit_dangerous_node
visit_InPlaceAssignmentNode
=
visit_dangerous_node
visit_Node
=
visit_safe_node
class
PyObjectTypeInferer
:
"""
...
...
@@ -175,7 +231,7 @@ class SimpleAssignmentTypeInferer:
entry
=
ready_to_infer
.
pop
()
types
=
[
expr
.
infer_type
(
scope
)
for
expr
in
entry
.
assignments
]
if
types
:
entry
.
type
=
spanning_type
(
types
)
entry
.
type
=
spanning_type
(
types
,
entry
.
might_overflow
)
else
:
# FIXME: raise a warning?
# print "No assignments", entry.pos, entry
...
...
@@ -188,9 +244,9 @@ class SimpleAssignmentTypeInferer:
if
len
(
deps
)
==
1
and
deps
==
set
([
entry
]):
types
=
[
expr
.
infer_type
(
scope
)
for
expr
in
entry
.
assignments
if
expr
.
type_dependencies
(
scope
)
==
()]
if
types
:
entry
.
type
=
spanning_type
(
types
)
entry
.
type
=
spanning_type
(
types
,
entry
.
might_overflow
)
types
=
[
expr
.
infer_type
(
scope
)
for
expr
in
entry
.
assignments
]
entry
.
type
=
spanning_type
(
types
)
# might be wider...
entry
.
type
=
spanning_type
(
types
,
entry
.
might_overflow
)
# might be wider...
resolve_dependancy
(
entry
)
del
dependancies_by_entry
[
entry
]
if
ready_to_infer
:
...
...
@@ -218,11 +274,11 @@ def find_spanning_type(type1, type2):
return
PyrexTypes
.
c_double_type
return
result_type
def
aggressive_spanning_type
(
types
):
def
aggressive_spanning_type
(
types
,
might_overflow
):
result_type
=
reduce
(
find_spanning_type
,
types
)
return
result_type
def
safe_spanning_type
(
types
):
def
safe_spanning_type
(
types
,
might_overflow
):
result_type
=
reduce
(
find_spanning_type
,
types
)
if
result_type
.
is_pyobject
:
# any specific Python type is always safe to infer
...
...
@@ -235,6 +291,22 @@ def safe_spanning_type(types):
# find_spanning_type() only returns 'bint' for clean boolean
# operations without other int types, so this is safe, too
return
result_type
elif
result_type
.
is_ptr
and
not
(
result_type
.
is_int
and
result_type
.
rank
==
0
):
# Any pointer except (signed|unsigned|) char* can't implicitly
# become a PyObject.
return
result_type
elif
result_type
.
is_cpp_class
:
# These can't implicitly become Python objects either.
return
result_type
elif
result_type
.
is_struct
:
# Though we have struct -> object for some structs, this is uncommonly
# used, won't arise in pure Python, and there shouldn't be side
# effects, so I'm declaring this safe.
return
result_type
# TODO: double complex should be OK as well, but we need
# to make sure everything is supported.
elif
result_type
.
is_int
and
not
might_overflow
:
return
result_type
return
py_object_type
...
...
Cython/Distutils/__init__.py
View file @
b978c7e0
...
...
@@ -7,5 +7,6 @@
# and keep the old one under the module name _build_ext,
# so that *our* build_ext can make use of it.
from
build_ext
import
build_ext
from
Cython.Distutils.build_ext
import
build_ext
# from extension import Extension
Cython/Distutils/build_ext.py
View file @
b978c7e0
...
...
@@ -15,16 +15,6 @@ from distutils.sysconfig import customize_compiler, get_python_version
from
distutils.dep_util
import
newer
,
newer_group
from
distutils
import
log
from
distutils.dir_util
import
mkpath
try
:
from
Cython.Compiler.Main
\
import
CompilationOptions
,
\
default_options
as
pyrex_default_options
,
\
compile
as
cython_compile
from
Cython.Compiler.Errors
import
PyrexError
except
ImportError
,
e
:
print
"failed to import Cython: %s"
%
e
PyrexError
=
None
from
distutils.command
import
build_ext
as
_build_ext
extension_name_re
=
_build_ext
.
extension_name_re
...
...
@@ -83,18 +73,22 @@ class build_ext(_build_ext.build_ext):
self
.
build_extension
(
ext
)
def
cython_sources
(
self
,
sources
,
extension
):
"""
Walk the list of source files in 'sources', looking for Cython
source files (.pyx and .py). Run Cython on all that are
found, and return a modified 'sources' list with Cython source
files replaced by the generated C (or C++) files.
"""
if
PyrexError
==
None
:
raise
DistutilsPlatformError
,
\
(
"Cython does not appear to be installed "
"on platform '%s'"
)
%
os
.
name
try
:
from
Cython.Compiler.Main
\
import
CompilationOptions
,
\
default_options
as
pyrex_default_options
,
\
compile
as
cython_compile
from
Cython.Compiler.Errors
import
PyrexError
except
ImportError
:
e
=
sys
.
exc_info
()[
1
]
print
(
"failed to import Cython: %s"
%
e
)
raise
DistutilsPlatformError
(
"Cython does not appear to be installed"
)
new_sources
=
[]
pyrex_sources
=
[]
...
...
Tools/cython-mode.el
View file @
b978c7e0
;;;; `Cython' mode. (add-to-list 'auto-mode-alist '("\\.pyx\\'" . cython-mode)) (define-derived-mode cython-mode python-mode "Cython" (font-lock-add-keywords nil `((,(concat "\\<\\(NULL" "\\|c\\(def\\|har\\|typedef\\)" "\\|e\\(num\\|xtern\\)" "\\|float" "\\|in\\(clude\\|t\\)" "\\|object\\|public\\|struct\\|type\\|union\\|void" "\\)\\>") 1 font-lock-keyword-face t))))
\ No newline at end of file
;; Cython mode
(
require
'python-mode
)
(
add-to-list
'auto-mode-alist
'
(
"\\.pyx\\'"
.
cython-mode
))
(
add-to-list
'auto-mode-alist
'
(
"\\.pxd\\'"
.
cython-mode
))
(
add-to-list
'auto-mode-alist
'
(
"\\.pxi\\'"
.
cython-mode
))
(
defun
cython-compile
()
"Compile the file via Cython."
(
interactive
)
(
let
((
cy-buffer
(
current-buffer
)))
(
with-current-buffer
(
compile
compile-command
)
(
set
(
make-local-variable
'cython-buffer
)
cy-buffer
)
(
add-to-list
(
make-local-variable
'compilation-finish-functions
)
'cython-compilation-finish
)))
)
(
defun
cython-compilation-finish
(
buffer
how
)
"Called when Cython compilation finishes."
;; XXX could annotate source here
)
(
defvar
cython-mode-map
(
let
((
map
(
make-sparse-keymap
)))
;; Will inherit from `python-mode-map' thanks to define-derived-mode.
(
define-key
map
"\C-c\C-c"
'cython-compile
)
map
)
"Keymap used in `cython-mode'."
)
(
defvar
cython-font-lock-keywords
`
(
;; new keywords in Cython language
(
,
(
regexp-opt
'
(
"by"
"cdef"
"cimport"
"cpdef"
"ctypedef"
"enum"
"except?"
"extern"
"gil"
"include"
"nogil"
"property"
"public"
"readonly"
"struct"
"union"
"DEF"
"IF"
"ELIF"
"ELSE"
)
'words
)
1
font-lock-keyword-face
)
;; C and Python types (highlight as builtins)
(
,
(
regexp-opt
'
(
"NULL"
"bint"
"char"
"dict"
"double"
"float"
"int"
"list"
"long"
"object"
"Py_ssize_t"
"short"
"size_t"
"void"
)
'words
)
1
font-lock-builtin-face
)
;; cdef is used for more than functions, so simply highlighting the next
;; word is problematic. struct, enum and property work though.
(
"\\<\\(?:struct\\|enum\\)[ \t]+\\([a-zA-Z_]+[a-zA-Z0-9_]*\\)"
1
py-class-name-face
)
(
"\\<property[ \t]+\\([a-zA-Z_]+[a-zA-Z0-9_]*\\)"
1
font-lock-function-name-face
))
"Additional font lock keywords for Cython mode."
)
(
define-derived-mode
cython-mode
python-mode
"Cython"
"Major mode for Cython development, derived from Python mode.
\\{cython-mode-map}"
(
setcar
font-lock-defaults
(
append
python-font-lock-keywords
cython-font-lock-keywords
))
(
set
(
make-local-variable
'compile-command
)
(
concat
"cython -a "
buffer-file-name
))
(
add-to-list
(
make-local-variable
'compilation-finish-functions
)
'cython-compilation-finish
)
)
(
provide
'cython-mode
)
runtests.py
View file @
b978c7e0
...
...
@@ -48,8 +48,8 @@ EXT_DEP_INCLUDES = [
]
VER_DEP_MODULES
=
{
# such as:
# (2,4) : (operator.le, lambda x: x in ['run.set'
]),
(
2
,
4
)
:
(
operator
.
le
,
lambda
x
:
x
in
[
'run.extern_builtins_T258'
]),
(
3
,):
(
operator
.
ge
,
lambda
x
:
x
in
[
'run.non_future_division'
,
'compile.extsetslice'
,
'compile.extdelslice'
]),
...
...
setup.py
View file @
b978c7e0
...
...
@@ -25,6 +25,11 @@ if sys.platform == "darwin":
setup_args
=
{}
def
add_command_class
(
name
,
cls
):
cmdclasses
=
setup_args
.
get
(
'cmdclass'
,
{})
cmdclasses
[
name
]
=
cls
setup_args
[
'cmdclass'
]
=
cmdclasses
if
sys
.
version_info
[
0
]
>=
3
:
import
lib2to3.refactor
from
distutils.command.build_py
\
...
...
@@ -34,7 +39,7 @@ if sys.version_info[0] >= 3:
if
fix
.
split
(
'fix_'
)[
-
1
]
not
in
(
'next'
,)
]
build_py
.
fixer_names
=
fixers
setup_args
[
'cmdclass'
]
=
{
"build_py"
:
build_py
}
add_command_class
(
"build_py"
,
build_py
)
if
sys
.
version_info
<
(
2
,
4
):
...
...
@@ -72,14 +77,45 @@ else:
else
:
scripts
=
[
"cython.py"
]
def
compile_cython_modules
():
source_root
=
os
.
path
.
abspath
(
os
.
path
.
dirname
(
__file__
))
compiled_modules
=
[
"Cython.Plex.Scanners"
,
"Cython.Compiler.Scanning"
,
"Cython.Compiler.Parsing"
,
"Cython.Compiler.Visitor"
,
"Cython.Runtime.refnanny"
]
extensions
=
[]
try
:
if
sys
.
version_info
[
0
]
>=
3
:
raise
ValueError
sys
.
argv
.
remove
(
"--no-cython-compile"
)
except
ValueError
:
try
:
from
Cython.Distutils
import
build_ext
as
build_ext_orig
for
module
in
compiled_modules
:
source_file
=
os
.
path
.
join
(
source_root
,
*
module
.
split
(
'.'
))
if
os
.
path
.
exists
(
source_file
+
".py"
):
pyx_source_file
=
source_file
+
".py"
else
:
pyx_source_file
=
source_file
+
".pyx"
extensions
.
append
(
Extension
(
module
,
sources
=
[
pyx_source_file
])
)
class
build_ext
(
build_ext_orig
):
def
build_extensions
(
self
):
# add path where 2to3 installed the transformed sources
# and make sure Python (re-)imports them from there
already_imported
=
[
module
for
module
in
sys
.
modules
if
module
==
'Cython'
or
module
.
startswith
(
'Cython.'
)
]
for
module
in
already_imported
:
del
sys
.
modules
[
module
]
sys
.
path
.
insert
(
0
,
os
.
path
.
join
(
source_root
,
self
.
build_lib
))
build_ext_orig
.
build_extensions
(
self
)
setup_args
[
'ext_modules'
]
=
extensions
add_command_class
(
"build_ext"
,
build_ext
)
else
:
# Python 2.x
from
distutils.command.build_ext
import
build_ext
as
build_ext_orig
try
:
class
build_ext
(
build_ext_orig
):
def
build_extension
(
self
,
ext
,
*
args
,
**
kargs
):
try
:
...
...
@@ -89,12 +125,6 @@ except ValueError:
from
Cython.Compiler.Main
import
compile
from
Cython
import
Utils
source_root
=
os
.
path
.
dirname
(
__file__
)
compiled_modules
=
[
"Cython.Plex.Scanners"
,
"Cython.Compiler.Scanning"
,
"Cython.Compiler.Parsing"
,
"Cython.Compiler.Visitor"
,
"Cython.Runtime.refnanny"
]
extensions
=
[]
for
module
in
compiled_modules
:
source_file
=
os
.
path
.
join
(
source_root
,
*
module
.
split
(
'.'
))
if
os
.
path
.
exists
(
source_file
+
".py"
):
...
...
@@ -116,11 +146,16 @@ except ValueError:
print
(
"Compilation failed"
)
if
extensions
:
setup_args
[
'ext_modules'
]
=
extensions
setup_args
[
'cmdclass'
]
=
{
"build_ext"
:
build_ext
}
add_command_class
(
"build_ext"
,
build_ext
)
except
Exception
:
print
(
"ERROR: %s"
%
sys
.
exc_info
()[
1
])
print
(
"Extension module compilation failed, using plain Python implementation"
)
try
:
sys
.
argv
.
remove
(
"--no-cython-compile"
)
except
ValueError
:
compile_cython_modules
()
setup_args
.
update
(
setuptools_extra_args
)
from
Cython.Compiler.Version
import
version
...
...
tests/compile/declarations.pyx
View file @
b978c7e0
...
...
@@ -11,6 +11,10 @@ cdef extern int (*iapfn())[5]
cdef
extern
char
*
(
*
cpapfn
())[
5
]
cdef
extern
int
fnargfn
(
int
())
cdef
extern
int
ia
[]
cdef
extern
int
iaa
[][
3
]
cdef
extern
int
a
(
int
[][
3
],
int
[][
3
][
5
])
cdef
void
f
():
cdef
void
*
p
=
NULL
global
ifnp
,
cpa
...
...
tests/compile/msvc_strings.pyx
0 → 100644
View file @
b978c7e0
This diff is collapsed.
Click to expand it.
tests/run/carray_slicing.pyx
View file @
b978c7e0
...
...
@@ -111,21 +111,84 @@ def slice_charptr_for_loop_c_enumerate():
############################################################
# tests for int* slicing
##
cdef int cints[6]
##
for i in range(6):
##
cints[i] = i
cdef
int
cints
[
6
]
for
i
in
range
(
6
):
cints
[
i
]
=
i
## @cython.test_assert_path_exists("//ForFromStatNode",
## "//ForFromStatNode//IndexNode")
## @cython.test_fail_if_path_exists("//ForInStatNode")
## def slice_intptr_for_loop_c():
## """
## >>> slice_intptr_for_loop_c()
## [0, 1, 2]
## [1, 2, 3, 4]
## [4, 5]
## """
## cdef int i
## print [ i for i in cints[:3] ]
## print [ i for i in cints[1:5] ]
## print [ i for i in cints[4:6] ]
@
cython
.
test_assert_path_exists
(
"//ForFromStatNode"
,
"//ForFromStatNode//IndexNode"
)
@
cython
.
test_fail_if_path_exists
(
"//ForInStatNode"
)
def
slice_intarray_for_loop_c
():
"""
>>> slice_intarray_for_loop_c()
[0, 1, 2]
[1, 2, 3, 4]
[4, 5]
"""
cdef
int
i
print
[
i
for
i
in
cints
[:
3
]
]
print
[
i
for
i
in
cints
[
1
:
5
]
]
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_intarray_for_loop_c
():
"""
>>> iter_intarray_for_loop_c()
[0, 1, 2, 3, 4, 5]
"""
cdef
int
i
print
[
i
for
i
in
cints
]
@
cython
.
test_assert_path_exists
(
"//ForFromStatNode"
,
"//ForFromStatNode//IndexNode"
)
@
cython
.
test_fail_if_path_exists
(
"//ForInStatNode"
)
def
slice_intptr_for_loop_c
():
"""
>>> slice_intptr_for_loop_c()
[0, 1, 2]
[1, 2, 3, 4]
[4, 5]
"""
cdef
int
*
nums
=
cints
cdef
int
i
print
[
i
for
i
in
nums
[:
3
]
]
print
[
i
for
i
in
nums
[
1
:
5
]
]
print
[
i
for
i
in
nums
[
4
:
6
]
]
############################################################
# tests for slicing other arrays
cdef
double
cdoubles
[
6
]
for
i
in
range
(
6
):
cdoubles
[
i
]
=
i
+
0.5
cdef
double
*
cdoubles_ptr
=
cdoubles
@
cython
.
test_assert_path_exists
(
"//ForFromStatNode"
,
"//ForFromStatNode//IndexNode"
)
@
cython
.
test_fail_if_path_exists
(
"//ForInStatNode"
)
def
slice_doublptr_for_loop_c
():
"""
>>> slice_doublptr_for_loop_c()
[0.5, 1.5, 2.5]
[1.5, 2.5, 3.5, 4.5]
[4.5, 5.5]
"""
cdef
double
d
print
[
d
for
d
in
cdoubles_ptr
[:
3
]
]
print
[
d
for
d
in
cdoubles_ptr
[
1
:
5
]
]
print
[
d
for
d
in
cdoubles_ptr
[
4
:
6
]
]
@
cython
.
test_assert_path_exists
(
"//ForFromStatNode"
,
"//ForFromStatNode//IndexNode"
)
@
cython
.
test_fail_if_path_exists
(
"//ForInStatNode"
)
def
iter_doublearray_for_loop_c
():
"""
>>> iter_doublearray_for_loop_c()
[0.5, 1.5, 2.5, 3.5, 4.5, 5.5]
"""
cdef
double
d
print
[
d
for
d
in
cdoubles
]
tests/run/complex_numbers_T305.pyx
View file @
b978c7e0
...
...
@@ -21,7 +21,7 @@ def test_arithmetic(double complex z, double complex w):
>>> test_arithmetic(5-10j, 3+4j)
((5-10j), (-5+10j), (8-6j), (2-14j), (55-10j), (-1-2j))
"""
return
+
z
,
-
z
,
z
+
w
,
z
-
w
,
z
*
w
,
z
/
w
return
+
z
,
-
z
+
0
,
z
+
w
,
z
-
w
,
z
*
w
,
z
/
w
@
cython
.
cdivision
(
False
)
def
test_div_by_zero
(
double
complex
z
):
...
...
tests/run/cpp_nested_templates.pyx
View file @
b978c7e0
...
...
@@ -21,11 +21,8 @@ def test_wrap_pair(int i, double x):
>>> test_wrap_pair(2, 2.25)
(2, 2.25, True)
"""
cdef
Pair
[
int
,
double
]
*
pair
cdef
Wrap
[
Pair
[
int
,
double
]]
*
wrap
try
:
pair
=
new
Pair
[
int
,
double
](
i
,
x
)
wrap
=
new
Wrap
[
Pair
[
int
,
double
]](
deref
(
pair
))
wrap
=
new
Wrap
[
Pair
[
int
,
double
]](
Pair
[
int
,
double
](
i
,
x
))
return
wrap
.
get
().
first
(),
wrap
.
get
().
second
(),
deref
(
wrap
)
==
deref
(
wrap
)
finally
:
del
pair
,
wrap
del
wrap
tests/run/cpp_templates.pyx
View file @
b978c7e0
...
...
@@ -21,7 +21,6 @@ def test_int(int x, int y):
>>> test_int(100, 100)
(100, 100, True)
"""
cdef
Wrap
[
int
]
*
a
,
*
b
try
:
a
=
new
Wrap
[
int
](
x
)
b
=
new
Wrap
[
int
](
0
)
...
...
@@ -38,7 +37,6 @@ def test_double(double x, double y):
>>> test_double(100, 100)
(100.0, 100.0, True)
"""
cdef
Wrap
[
double
]
*
a
,
*
b
try
:
a
=
new
Wrap
[
double
](
x
)
b
=
new
Wrap
[
double
](
-
1
)
...
...
@@ -54,7 +52,6 @@ def test_pair(int i, double x):
>>> test_pair(2, 2.25)
(2, 2.25, True, False)
"""
cdef
Pair
[
int
,
double
]
*
pair
try
:
pair
=
new
Pair
[
int
,
double
](
i
,
x
)
return
pair
.
first
(),
pair
.
second
(),
deref
(
pair
)
==
deref
(
pair
),
deref
(
pair
)
!=
deref
(
pair
)
...
...
@@ -68,7 +65,6 @@ def test_ptr(int i):
>>> test_ptr(5)
5
"""
cdef
Wrap
[
int
*
]
*
w
try
:
w
=
new
Wrap
[
int
*
](
&
i
)
return
deref
(
w
.
get
())
...
...
@@ -85,7 +81,6 @@ def test_func_ptr(double x):
>>> test_func_ptr(-1.5)
2.25
"""
cdef
Wrap
[
double
(
*
)(
double
)]
*
w
try
:
w
=
new
Wrap
[
double
(
*
)(
double
)](
&
f
)
return
w
.
get
()(
x
)
...
...
tests/run/extern_builtins_T258.pyx
View file @
b978c7e0
cdef
extern
from
"Python.h"
:
ctypedef
class
__builtin__
.
str
[
object
PyStringObject
]:
cdef
long
ob_shash
ctypedef
class
__builtin__
.
list
[
object
PyListObject
]:
cdef
Py_ssize_t
ob_size
cdef
Py_ssize_t
allocated
ctypedef
class
__builtin__
.
dict
[
object
PyDictObject
]:
pass
cdef
str
s
=
"abc"
cdef
Py_ssize_t
Py_SIZE
(
object
o
)
cdef
list
L
=
[
1
,
2
,
4
]
cdef
dict
d
=
{
'A'
:
'a'
}
...
...
@@ -23,18 +20,7 @@ def test_list(list L):
>>> test_list(list_subclass([1,2,3]))
True
"""
return
L
.
ob_size
<=
L
.
allocated
def
test_str
(
str
s
):
"""
>>> test_str("abc")
True
>>> class str_subclass(str): pass
>>> test_str(str_subclass("xyz"))
True
"""
cdef
char
*
ss
=
s
return
hash
(
s
)
==
s
.
ob_shash
return
Py_SIZE
(
L
)
<=
L
.
allocated
def
test_tuple
(
tuple
t
):
"""
...
...
tests/run/type_inference.pyx
View file @
b978c7e0
...
...
@@ -93,13 +93,49 @@ def arithmetic():
>>> arithmetic()
"""
a
=
1
+
2
assert
typeof
(
a
)
==
"long"
assert
typeof
(
a
)
==
"long"
,
typeof
(
a
)
b
=
1
+
1.5
assert
typeof
(
b
)
==
"double"
assert
typeof
(
b
)
==
"double"
,
typeof
(
b
)
c
=
1
+
<
object
>
2
assert
typeof
(
c
)
==
"Python object"
d
=
"abc %s"
%
"x"
assert
typeof
(
d
)
==
"Python object"
assert
typeof
(
c
)
==
"Python object"
,
typeof
(
c
)
d
=
1
*
1.5
**
2
assert
typeof
(
d
)
==
"double"
,
typeof
(
d
)
def
builtin_type_operations
():
"""
>>> builtin_type_operations()
"""
b1
=
b'a'
*
10
b1
=
10
*
b'a'
b1
=
10
*
b'a'
*
10
assert
typeof
(
b1
)
==
"bytes object"
,
typeof
(
b1
)
b2
=
b'a'
+
b'b'
assert
typeof
(
b2
)
==
"bytes object"
,
typeof
(
b2
)
u1
=
u'a'
*
10
u1
=
10
*
u'a'
assert
typeof
(
u1
)
==
"unicode object"
,
typeof
(
u1
)
u2
=
u'a'
+
u'b'
assert
typeof
(
u2
)
==
"unicode object"
,
typeof
(
u2
)
u3
=
u'a%s'
%
u'b'
u3
=
u'a%s'
%
10
assert
typeof
(
u3
)
==
"unicode object"
,
typeof
(
u3
)
s1
=
"abc %s"
%
"x"
s1
=
"abc %s"
%
10
assert
typeof
(
s1
)
==
"str object"
,
typeof
(
s1
)
s2
=
"abc %s"
+
"x"
assert
typeof
(
s2
)
==
"str object"
,
typeof
(
s2
)
s3
=
"abc %s"
*
10
s3
=
"abc %s"
*
10
*
10
s3
=
10
*
"abc %s"
*
10
assert
typeof
(
s3
)
==
"str object"
,
typeof
(
s3
)
L1
=
[]
+
[]
assert
typeof
(
L1
)
==
"list object"
,
typeof
(
L1
)
L2
=
[]
*
2
assert
typeof
(
L2
)
==
"list object"
,
typeof
(
L2
)
T1
=
()
+
()
assert
typeof
(
T1
)
==
"tuple object"
,
typeof
(
T1
)
T2
=
()
*
2
assert
typeof
(
T2
)
==
"tuple object"
,
typeof
(
T2
)
def
cascade
():
"""
...
...
@@ -215,10 +251,29 @@ def safe_only():
"""
a
=
1.0
assert
typeof
(
a
)
==
"double"
,
typeof
(
c
)
b
=
1
assert
typeof
(
b
)
==
"
Python object
"
,
typeof
(
b
)
b
=
1
;
assert
typeof
(
b
)
==
"
long
"
,
typeof
(
b
)
c
=
MyType
()
assert
typeof
(
c
)
==
"MyType"
,
typeof
(
c
)
for
i
in
range
(
10
):
pass
assert
typeof
(
i
)
==
"long"
,
typeof
(
i
)
d
=
1
res
=
~
d
assert
typeof
(
d
)
==
"long"
,
typeof
(
d
)
# potentially overflowing arithmatic
e
=
1
e
+=
1
assert
typeof
(
e
)
==
"Python object"
,
typeof
(
e
)
f
=
1
res
=
f
*
10
assert
typeof
(
f
)
==
"Python object"
,
typeof
(
f
)
g
=
1
res
=
10
*
(
~
g
)
assert
typeof
(
g
)
==
"Python object"
,
typeof
(
g
)
for
j
in
range
(
10
):
res
=
-
j
assert
typeof
(
j
)
==
"Python object"
,
typeof
(
j
)
@
infer_types
(
None
)
def
args_tuple_keywords
(
*
args
,
**
kwargs
):
...
...
@@ -249,3 +304,36 @@ def args_tuple_keywords_reassign_pyobjects(*args, **kwargs):
args
=
[]
kwargs
=
"test"
# / A -> AA -> AAA
# Base0 -> Base -
# \ B -> BB
# C -> CC
cdef
class
Base0
:
pass
cdef
class
Base
(
Base0
):
pass
cdef
class
A
(
Base
):
pass
cdef
class
AA
(
A
):
pass
cdef
class
AAA
(
AA
):
pass
cdef
class
B
(
Base
):
pass
cdef
class
BB
(
B
):
pass
cdef
class
C
:
pass
cdef
class
CC
(
C
):
pass
@
infer_types
(
None
)
def
common_extension_type_base
():
"""
>>> common_extension_type_base()
"""
x
=
A
()
x
=
AA
()
assert
typeof
(
x
)
==
"A"
,
typeof
(
x
)
y
=
A
()
y
=
B
()
assert
typeof
(
y
)
==
"Base"
,
typeof
(
y
)
z
=
AAA
()
z
=
BB
()
assert
typeof
(
z
)
==
"Base"
,
typeof
(
z
)
w
=
A
()
w
=
CC
()
assert
typeof
(
w
)
==
"Python object"
,
typeof
(
w
)
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