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
dbb891df
Commit
dbb891df
authored
Jan 15, 2007
by
Robert Bradshaw
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Implemented inplace arithmetic
parent
e8d25746
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
127 additions
and
5 deletions
+127
-5
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+16
-2
Cython/Compiler/Lexicon.py
Cython/Compiler/Lexicon.py
+1
-1
Cython/Compiler/Nodes.py
Cython/Compiler/Nodes.py
+98
-0
Cython/Compiler/Parsing.py
Cython/Compiler/Parsing.py
+12
-2
No files found.
Cython/Compiler/ExprNodes.py
View file @
dbb891df
...
...
@@ -3041,8 +3041,11 @@ class CloneNode(CoercionNode):
def
__init__
(
self
,
arg
):
CoercionNode
.
__init__
(
self
,
arg
)
if
hasattr
(
arg
,
'type'
):
self
.
type
=
arg
.
type
self
.
result_ctype
=
arg
.
result_ctype
if
hasattr
(
arg
,
'entry'
):
self
.
entry
=
arg
.
entry
def
calculate_result_code
(
self
):
return
self
.
arg
.
result_code
...
...
@@ -3051,6 +3054,8 @@ class CloneNode(CoercionNode):
self
.
type
=
self
.
arg
.
type
self
.
result_ctype
=
self
.
arg
.
result_ctype
self
.
is_temp
=
1
if
hasattr
(
self
.
arg
,
'entry'
):
self
.
entry
=
self
.
arg
.
entry
#def result_as_extension_type(self):
# return self.arg.result_as_extension_type()
...
...
@@ -3061,6 +3066,15 @@ class CloneNode(CoercionNode):
def
generate_result_code
(
self
,
code
):
pass
def
generate_disposal_code
(
self
,
code
):
code
.
putln
(
"// ---- CloneNode.generate_disposal_code() for %s"
%
self
.
arg
.
result_code
)
def
allocate_temps
(
self
,
env
):
self
.
result_code
=
self
.
calculate_result_code
()
def
release_temp
(
self
,
env
):
pass
#------------------------------------------------------------------------------------
#
# Runtime support code
...
...
Cython/Compiler/Lexicon.py
View file @
dbb891df
...
...
@@ -66,7 +66,7 @@ def make_lexicon():
bra
=
Any
(
"([{"
)
ket
=
Any
(
")]}"
)
punct
=
Any
(
":,;+-*/|&<>=.%`~^?"
)
diphthong
=
Str
(
"=="
,
"<>"
,
"!="
,
"<="
,
">="
,
"<<"
,
">>"
,
"**"
)
diphthong
=
Str
(
"=="
,
"<>"
,
"!="
,
"<="
,
">="
,
"<<"
,
">>"
,
"**"
,
"+="
,
"-="
,
"*="
,
"/="
,
"%="
,
"|="
,
"^="
,
"&="
)
spaces
=
Rep1
(
Any
(
"
\
t
\
f
"
))
comment
=
Str
(
"#"
)
+
Rep
(
AnyBut
(
"
\
n
"
))
escaped_newline
=
Str
(
"
\
\
\
n
"
)
...
...
Cython/Compiler/Nodes.py
View file @
dbb891df
...
...
@@ -2708,6 +2708,104 @@ class ParallelAssignmentNode(AssignmentNode):
for
stat
in
self
.
stats
:
stat
.
generate_assignment_code
(
code
)
class
InPlaceAssignmentNode
(
AssignmentNode
):
# An in place arithmatic operand:
#
# a += b
# a -= b
# ...
#
# lhs ExprNode Left hand side
# rhs ExprNode Right hand side
# op char one of "+-*/%^&|"
# dup (ExprNode) copy of lhs used for operation (auto-generated)
#
# This code is a bit tricky because in order to obey Python
# semantics the sub-expressions (e.g. indices) of the lhs must
# not be evaluated twice. So we must re-use the values calculated
# in evaluation phase for the assignment phase as well.
# Fortunately, the type of the lhs node is fairly constrained
# (it must be a NameNode, AttributeNode, or IndexNode).
def
analyse_declarations
(
self
,
env
):
self
.
lhs
.
analyse_target_declaration
(
env
)
def
analyse_expressions_1
(
self
,
env
,
use_temp
=
0
):
import
ExprNodes
self
.
create_dup_node
(
env
)
# re-assigns lhs to a shallow copy
self
.
rhs
.
analyse_types
(
env
)
self
.
lhs
.
analyse_target_types
(
env
)
if
self
.
lhs
.
type
.
is_pyobject
or
self
.
rhs
.
type
.
is_pyobject
:
self
.
rhs
=
self
.
rhs
.
coerce_to
(
self
.
lhs
.
type
,
env
)
if
self
.
lhs
.
type
.
is_pyobject
:
self
.
result
=
ExprNodes
.
PyTempNode
(
self
.
pos
,
env
)
self
.
result
.
allocate_temps
(
env
)
if
use_temp
:
self
.
rhs
=
self
.
rhs
.
coerce_to_temp
(
env
)
self
.
dup
.
allocate_subexpr_temps
(
env
)
self
.
dup
.
allocate_temp
(
env
)
self
.
rhs
.
allocate_temps
(
env
)
def
analyse_expressions_2
(
self
,
env
):
self
.
lhs
.
allocate_target_temps
(
env
)
self
.
lhs
.
release_target_temp
(
env
)
self
.
dup
.
release_temp
(
env
)
if
self
.
dup
.
is_temp
:
self
.
dup
.
release_subexpr_temps
(
env
)
self
.
rhs
.
release_temp
(
env
)
if
self
.
lhs
.
type
.
is_pyobject
:
self
.
result
.
release_temp
(
env
)
def
generate_execution_code
(
self
,
code
):
self
.
rhs
.
generate_evaluation_code
(
code
)
self
.
dup
.
generate_subexpr_evaluation_code
(
code
)
self
.
dup
.
generate_result_code
(
code
)
if
self
.
lhs
.
type
.
is_pyobject
:
code
.
putln
(
"//---- iadd code"
);
code
.
putln
(
"%s = %s(%s, %s); if (!%s) %s"
%
(
self
.
result
.
result_code
,
self
.
py_operation_function
(),
self
.
dup
.
py_result
(),
self
.
rhs
.
py_result
(),
self
.
result
.
py_result
(),
code
.
error_goto
(
self
.
pos
)))
self
.
rhs
.
generate_disposal_code
(
code
)
self
.
dup
.
generate_disposal_code
(
code
)
self
.
lhs
.
generate_assignment_code
(
self
.
result
,
code
)
else
:
# have to do assignment directly to avoid side-effects
code
.
putln
(
"%s %s= %s;"
%
(
self
.
lhs
.
result_code
,
self
.
operator
,
self
.
rhs
.
result_code
)
)
self
.
rhs
.
generate_disposal_code
(
code
)
if
self
.
dup
.
is_temp
:
self
.
dup
.
generate_subexpr_disposal_code
(
code
)
def
create_dup_node
(
self
,
env
):
import
ExprNodes
self
.
dup
=
self
.
lhs
self
.
dup
.
analyse_types
(
env
)
if
isinstance
(
self
.
lhs
,
ExprNodes
.
NameNode
):
target_lhs
=
ExprNodes
.
NameNode
(
self
.
dup
.
pos
,
name
=
self
.
dup
.
name
,
is_temp
=
self
.
dup
.
is_temp
,
entry
=
self
.
dup
.
entry
)
elif
isinstance
(
self
.
lhs
,
ExprNodes
.
AttributeNode
):
target_lhs
=
ExprNodes
.
AttributeNode
(
self
.
dup
.
pos
,
obj
=
ExprNodes
.
CloneNode
(
self
.
lhs
.
obj
),
attribute
=
self
.
dup
.
attribute
,
is_temp
=
self
.
dup
.
is_temp
)
elif
isinstance
(
self
.
lhs
,
ExprNodes
.
IndexNode
):
target_lhs
=
ExprNodes
.
IndexNode
(
self
.
dup
.
pos
,
base
=
ExprNodes
.
CloneNode
(
self
.
dup
.
base
),
index
=
ExprNodes
.
CloneNode
(
self
.
lhs
.
index
),
is_temp
=
self
.
dup
.
is_temp
)
self
.
lhs
=
target_lhs
def
py_operation_function
(
self
):
return
self
.
py_functions
[
self
.
operator
]
py_functions
=
{
"|"
:
"PyNumber_InPlaceOr"
,
"^"
:
"PyNumber_InPlaceXor"
,
"&"
:
"PyNumber_InPlaceAnd"
,
"+"
:
"PyNumber_InPlaceAdd"
,
"-"
:
"PyNumber_InPlaceSubtract"
,
"*"
:
"PyNumber_InPlaceMultiply"
,
"/"
:
"PyNumber_InPlaceDivide"
,
"%"
:
"PyNumber_InPlaceRemainder"
,
}
class
PrintStatNode
(
StatNode
):
# print statement
...
...
Cython/Compiler/Parsing.py
View file @
dbb891df
...
...
@@ -700,6 +700,15 @@ def p_expression_or_assignment(s):
s
.
next
()
expr_list
.
append
(
p_expr
(
s
))
if
len
(
expr_list
)
==
1
:
if
re
.
match
(
"[+*/
\
%^
\
&|-]="
,
s
.
sy
):
lhs
=
expr_list
[
0
]
if
not
isinstance
(
lhs
,
(
ExprNodes
.
AttributeNode
,
ExprNodes
.
IndexNode
,
ExprNodes
.
NameNode
)
):
error
(
lhs
.
pos
,
"Illegal operand for inplace operation."
)
operator
=
s
.
sy
[
0
]
s
.
next
()
rhs
=
p_expr
(
s
)
return
Nodes
.
InPlaceAssignmentNode
(
lhs
.
pos
,
operator
=
operator
,
lhs
=
lhs
,
rhs
=
rhs
)
else
:
expr
=
expr_list
[
0
]
return
Nodes
.
ExprStatNode
(
expr
.
pos
,
expr
=
expr
)
else
:
...
...
@@ -1835,6 +1844,7 @@ def p_module(s, pxd, full_module_name):
#----------------------------------------------
def
print_parse_tree
(
f
,
node
,
level
,
key
=
None
):
from
Nodes
import
Node
ind
=
" "
*
level
if
node
:
f
.
write
(
ind
)
...
...
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