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
5061256f
Commit
5061256f
authored
Apr 30, 2011
by
Mark Florisson
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Constant fold fused type check expressions with branch pruning
parent
e3d28878
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
202 additions
and
57 deletions
+202
-57
Cython/Compiler/Nodes.py
Cython/Compiler/Nodes.py
+20
-5
Cython/Compiler/Optimize.py
Cython/Compiler/Optimize.py
+5
-1
Cython/Compiler/ParseTreeTransforms.py
Cython/Compiler/ParseTreeTransforms.py
+81
-35
Cython/Compiler/PyrexTypes.py
Cython/Compiler/PyrexTypes.py
+32
-16
tests/run/check_fused_types.pyx
tests/run/check_fused_types.pyx
+64
-0
No files found.
Cython/Compiler/Nodes.py
View file @
5061256f
...
...
@@ -29,6 +29,7 @@ from StringEncoding import EncodedString, escape_byte_string, split_string_liter
import
Options
import
ControlFlow
import
DebugFlags
from
Cython.Compiler
import
Errors
absolute_path_length
=
0
...
...
@@ -942,14 +943,23 @@ class FusedTypeNode(CBaseTypeNode):
return
self
.
types
[
0
]
types
=
[]
seen
=
cython
.
set
()
for
type
in
self
.
types
:
self
.
add_type
(
type
,
types
,
seen
)
return
PyrexTypes
.
FusedType
(
types
)
def
add_type
(
self
,
type
,
types
,
seen
):
if
type
not
in
seen
:
seen
.
add
(
type
)
if
type
.
is_fused
:
types
.
extend
(
type
.
types
)
for
specific_type
in
PyrexTypes
.
get_specific_types
(
type
):
self
.
add_type
(
specific_type
,
types
,
seen
)
else
:
types
.
append
(
type
)
return
PyrexTypes
.
FusedType
(
types
)
class
CVarDefNode
(
StatNode
):
# C variable definition or forward/extern function declaration.
...
...
@@ -2055,8 +2065,13 @@ class FusedCFuncDefNode(StatListNode):
cname
=
self
.
node
.
type
.
get_specific_cname
(
cname
)
copied_node
.
entry
.
func_cname
=
copied_node
.
entry
.
cname
=
cname
# TransformBuiltinMethods(copied_node)
ParseTreeTransforms
.
ReplaceFusedTypeChecks
(
copied_node
.
local_scope
)(
copied_node
)
num_errors
=
Errors
.
num_errors
transform
=
ParseTreeTransforms
.
ReplaceFusedTypeChecks
(
copied_node
.
local_scope
)
transform
(
copied_node
)
if
Errors
.
num_errors
>
num_errors
:
break
class
PyArgDeclNode
(
Node
):
...
...
Cython/Compiler/Optimize.py
View file @
5061256f
...
...
@@ -2974,8 +2974,12 @@ class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations):
literal nodes at each step. Non-literal nodes are never merged
into a single node.
"""
check_constant_value_not_set
=
True
def
_calculate_const
(
self
,
node
):
if
node
.
constant_result
is
not
ExprNodes
.
constant_value_not_set
:
if
(
self
.
check_constant_value_not_set
and
node
.
constant_result
is
not
ExprNodes
.
constant_value_not_set
):
return
# make sure we always set the value
...
...
Cython/Compiler/ParseTreeTransforms.py
View file @
5061256f
...
...
@@ -1899,48 +1899,94 @@ class ReplaceFusedTypeChecks(VisitorTransform):
self
.
local_scope
=
local_scope
def
visit_IfStatNode
(
self
,
node
):
if_clauses
=
node
.
if_clauses
[:]
"""
Filters out any if clauses with false compile time type check
expression.
"""
from
Cython.Compiler
import
Optimize
self
.
visitchildren
(
node
)
if
if_clauses
!=
node
.
if_clauses
:
if
node
.
if_clauses
:
return
node
.
if_clauses
[
0
]
return
node
.
else_clause
if_clauses
=
[]
seen_true
=
False
seen_true_anywhere
=
False
for
if_clause
in
node
.
if_clauses
:
transform
=
Optimize
.
ConstantFolding
()
transform
.
check_constant_value_not_set
=
False
transform
(
if_clause
.
condition
)
is_const
=
if_clause
.
condition
.
has_constant_result
()
const
=
if_clause
.
condition
.
constant_result
if
is_const
:
if
const
and
not
seen_true
:
seen_true
=
True
seen_true_anywhere
=
True
if_clauses
.
append
(
if_clause
)
else
:
seen_true
=
False
if_clauses
.
append
(
if_clause
)
if
if_clauses
:
node
.
if_clauses
=
if_clauses
if
seen_true_anywhere
:
node
.
else_clause
=
None
else
:
node
=
node
.
else_clause
return
node
def
visit_IfClauseNode
(
self
,
node
):
cond
=
node
.
condition
if
isinstance
(
cond
,
ExprNodes
.
PrimaryCmpNode
):
type1
=
cond
.
operand1
.
analyse_as_type
(
self
.
local_scope
)
type2
=
cond
.
operand2
.
analyse_as_type
(
self
.
local_scope
)
if
type1
and
type2
:
type1
=
self
.
specialize_type
(
type1
,
cond
.
operand1
.
pos
)
op
=
cond
.
operator
if
op
==
'is'
:
type2
=
self
.
specialize_type
(
type2
,
cond
.
operand1
.
pos
)
if
type1
.
same_as
(
type2
):
return
node
.
body
elif
op
in
(
'in'
,
'not_in'
):
# We have to do an instance check directly, as operand2
# needs to be a fused type and not a type with a subtype
# that is fused. First unpack the typedef
if
isinstance
(
type2
,
PyrexTypes
.
CTypedefType
):
type2
=
type2
.
typedef_base_type
if
type1
.
is_fused
or
not
isinstance
(
type2
,
PyrexTypes
.
FusedType
):
error
(
cond
.
pos
,
"Can use 'in' or 'not in' only on a "
"specific and a fused type"
)
elif
op
==
'in'
:
if
type1
in
type2
.
types
:
return
node
.
body
def
visit_PrimaryCmpNode
(
self
,
node
):
type1
=
node
.
operand1
.
analyse_as_type
(
self
.
local_scope
)
type2
=
node
.
operand2
.
analyse_as_type
(
self
.
local_scope
)
if
type1
and
type2
:
false
=
ExprNodes
.
BoolNode
(
node
.
pos
,
value
=
False
)
true
=
ExprNodes
.
BoolNode
(
node
.
pos
,
value
=
True
)
type1
=
self
.
specialize_type
(
type1
,
node
.
operand1
.
pos
)
op
=
node
.
operator
if
op
in
(
'is'
,
'is not'
,
'=='
,
'!='
):
type2
=
self
.
specialize_type
(
type2
,
node
.
operand2
.
pos
)
is_same
=
type1
.
same_as
(
type2
)
eq
=
op
in
(
'is'
,
'=='
)
if
(
is_same
and
eq
)
or
(
not
is_same
and
not
eq
):
return
true
elif
op
in
(
'in'
,
'not_in'
):
# We have to do an instance check directly, as operand2
# needs to be a fused type and not a type with a subtype
# that is fused. First unpack the typedef
if
isinstance
(
type2
,
PyrexTypes
.
CTypedefType
):
type2
=
type2
.
typedef_base_type
if
type1
.
is_fused
:
error
(
node
.
operand1
.
pos
,
"Type is fused"
)
elif
not
type2
.
is_fused
:
error
(
node
.
operand2
.
pos
,
"Can only use 'in' or 'not in' on a fused type"
)
else
:
if
not
isinstance
(
type2
,
PyrexTypes
.
FusedType
):
# Composed fused type, get all specific versions
types
=
PyrexTypes
.
get_specific_types
(
type2
)
else
:
if
type1
not
in
type2
.
types
:
return
node
.
body
types
=
type2
.
types
return
None
for
specific_type
in
types
:
if
type1
.
same_as
(
specific_type
):
if
op
==
'in'
:
return
true
else
:
return
false
if
op
==
'not_in'
:
return
true
return
false
return
node
...
...
Cython/Compiler/PyrexTypes.py
View file @
5061256f
...
...
@@ -2023,18 +2023,7 @@ class CFuncType(CType):
if
fused_types
is
None
:
fused_types
=
self
.
get_fused_types
()
fused_type
=
fused_types
[
0
]
for
specific_type
in
fused_type
.
types
:
cname
=
str
(
specific_type
)
result_fused_to_specific
=
{
fused_type
:
specific_type
}
if
len
(
fused_types
)
>
1
:
it
=
self
.
get_all_specific_permutations
(
fused_types
[
1
:])
for
next_cname
,
fused_to_specific
in
it
:
d
=
dict
(
fused_to_specific
,
**
result_fused_to_specific
)
yield
'%s_%s'
%
(
cname
,
next_cname
),
d
else
:
yield
cname
,
result_fused_to_specific
return
get_all_specific_permutations
(
fused_types
)
def
get_all_specific_function_types
(
self
):
"""
...
...
@@ -2042,23 +2031,25 @@ class CFuncType(CType):
"""
assert
self
.
is_fused
result
=
[]
permutations
=
self
.
get_all_specific_permutations
()
for
cname
,
fused_to_specific
in
permutations
:
new_func_type
=
self
.
entry
.
type
.
specialize
(
fused_to_specific
)
new_entry
=
copy
.
deepcopy
(
self
.
entry
)
new_entry
.
cname
=
self
.
get_specific_cname
(
cname
)
new_entry
.
type
=
new_func_type
new_entry
.
type
=
new_func_type
new_func_type
.
entry
=
new_entry
yield
new_func_type
result
.
append
(
new_func_type
)
return
result
def
get_specific_cname
(
self
,
fused_cname
):
"""
Given the cname for a permutation of fused types, return the cname
for the corresponding function with specific types.
The fused_cname is usually '_'.join(str(t) for t in specific_types)
"""
assert
self
.
is_fused
return
'%s%s%s'
%
(
Naming
.
fused_func_prefix
,
...
...
@@ -2086,6 +2077,31 @@ class CFuncType(CType):
# a normal cdef
return
func
(
entry
,
*
args
,
**
kwargs
)
def
get_all_specific_permutations
(
fused_types
,
id
=
"0"
,
f2s
=
()):
fused_type
=
fused_types
[
0
]
result
=
[]
for
newid
,
specific_type
in
enumerate
(
fused_type
.
types
):
f2s
=
dict
(
f2s
,
**
{
fused_type
:
specific_type
})
cname
=
'%s_%s'
%
(
id
,
newid
)
if
len
(
fused_types
)
>
1
:
result
.
extend
(
get_all_specific_permutations
(
fused_types
[
1
:],
cname
,
f2s
))
else
:
result
.
append
((
cname
,
f2s
))
return
result
def
get_specific_types
(
type
):
assert
type
.
is_fused
result
=
[]
for
cname
,
f2s
in
get_all_specific_permutations
(
type
.
get_fused_types
()):
result
.
append
(
type
.
specialize
(
f2s
))
return
result
class
CFuncTypeArg
(
BaseType
):
# name string
...
...
tests/run/check_fused_types.pyx
View file @
5061256f
...
...
@@ -2,8 +2,16 @@ cimport cython
cimport
check_fused_types_pxd
ctypedef
char
*
string_t
ctypedef
cython
.
fused_type
(
int
,
long
,
float
,
string_t
)
fused_t
ctypedef
cython
.
fused_type
(
int
,
long
)
other_t
ctypedef
cython
.
fused_type
(
short
,
short
int
,
short
,
int
)
base_t
ctypedef
cython
.
fused_type
(
float
complex
,
double
complex
,
int
complex
,
long
complex
)
complex_t
ctypedef
base_t
**
base_t_p_p
ctypedef
cython
.
fused_type
(
char
,
base_t_p_p
,
fused_t
,
complex_t
)
composed_t
cdef
func
(
fused_t
a
,
other_t
b
):
cdef
int
int_a
...
...
@@ -121,3 +129,59 @@ def test_if_then_else_float_int():
cdef
int
y
=
1
if_then_else
(
x
,
y
)
cdef
composed_t
composed
(
composed_t
x
,
composed_t
y
):
if
composed_t
in
base_t_p_p
or
composed_t
is
string_t
:
if
string_t
==
composed_t
:
print
x
,
y
else
:
print
x
[
0
][
0
],
y
[
0
][
0
]
return
x
elif
composed_t
==
string_t
:
print
'this is never executed'
elif
list
():
print
'neither is this one'
else
:
if
composed_t
not
in
complex_t
:
print
'not a complex number'
print
<
int
>
x
,
<
int
>
y
else
:
print
'it is a complex number'
print
x
.
real
,
x
.
imag
return
x
+
y
def
test_composed_types
():
"""
>>> test_composed_types()
it is a complex number
0.5 0.6
(0.9+0.4j)
<BLANKLINE>
not a complex number
9 10
19
<BLANKLINE>
7 8
<BLANKLINE>
spam eggs
spam
"""
cdef
double
complex
a
=
0.5
+
0.6j
,
b
=
0.4
-
0.2j
,
result
cdef
int
c
=
7
,
d
=
8
cdef
int
*
cp
=
&
c
,
*
dp
=
&
d
cdef
string_t
e
=
"spam"
,
f
=
"eggs"
result
=
composed
(
a
,
b
)
print
result
print
print
composed
(
c
+
2
,
d
+
2
)
print
composed
(
&
cp
,
&
dp
)
print
print
composed
(
e
,
f
)
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