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
c2ca0663
Commit
c2ca0663
authored
May 16, 2008
by
Dag Sverre Seljebotn
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
VisitorTransform + smaller Transform changes
parent
02218626
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
117 additions
and
25 deletions
+117
-25
Cython/Compiler/Transform.py
Cython/Compiler/Transform.py
+117
-25
No files found.
Cython/Compiler/Transform.py
View file @
c2ca0663
...
...
@@ -3,11 +3,22 @@
#
import
Nodes
import
ExprNodes
import
inspect
class
Transform
(
object
):
# parent_stack [Node] A stack providing information about where in the tree
# we currently are. Nodes here should be considered
# read-only.
# parent_stack [Node] A stack providing information about where in the tree
# we currently are. Nodes here should be considered
# read-only.
#
# attr_stack [(string,int|None)]
# A stack providing information about the attribute names
# followed to get to the current location in the tree.
# The first tuple item is the attribute name, the second is
# the index if the attribute is a list, or None otherwise.
#
#
# Additionally, any keyword arguments to __call__ will be set as fields while in
# a transformation.
# Transforms for the parse tree should usually extend this class for convenience.
# The caller of a transform will only first call initialize and then process_node on
...
...
@@ -18,12 +29,6 @@ class Transform(object):
# return the input node untouched. Returning None will remove the node from the
# parent.
def
__init__
(
self
):
self
.
parent_stack
=
[]
def
initialize
(
self
,
phase
,
**
options
):
pass
def
process_children
(
self
,
node
):
"""For all children of node, either process_list (if isinstance(node, list))
or process_node (otherwise) is called."""
...
...
@@ -36,29 +41,103 @@ class Transform(object):
newchild
=
self
.
process_list
(
child
,
childacc
.
name
())
if
not
isinstance
(
newchild
,
list
):
raise
Exception
(
"Cannot replace list with non-list!"
)
else
:
newchild
=
self
.
process_node
(
child
,
childacc
.
name
())
self
.
attr_stack
.
append
((
childacc
.
name
(),
None
))
newchild
=
self
.
process_node
(
child
)
if
newchild
is
not
None
and
not
isinstance
(
newchild
,
Nodes
.
Node
):
raise
Exception
(
"Cannot replace Node with non-Node!"
)
self
.
attr_stack
.
pop
()
childacc
.
set
(
newchild
)
self
.
parent_stack
.
pop
()
def
process_list
(
self
,
l
,
name
):
"""Calls process_node on all the items in l, using the name one gets when appending
[idx] to the name. Each item in l is transformed in-place by the item process_node
returns, then l is returned."""
# Comment: If moving to a copying strategy, it might makes sense to return a
# new list instead.
def
process_list
(
self
,
l
,
attrname
):
"""Calls process_node on all the items in l. Each item in l is transformed
in-place by the item process_node returns, then l is returned. If process_node
returns None, the item is removed from the list."""
for
idx
in
xrange
(
len
(
l
)):
l
[
idx
]
=
self
.
process_node
(
l
[
idx
],
"%s[%d]"
%
(
name
,
idx
))
return
l
self
.
attr_stack
.
append
((
attrname
,
idx
))
l
[
idx
]
=
self
.
process_node
(
l
[
idx
])
self
.
attr_stack
.
pop
()
return
[
x
for
x
in
l
if
x
is
not
None
]
def
process_node
(
self
,
node
,
name
):
def
process_node
(
self
,
node
):
"""Override this method to process nodes. name specifies which kind of relation the
parent has with child. This method should always return the node which the parent
should use for this relation, which can either be the same node, None to remove
the node, or a different node."""
raise
NotImplementedError
(
"Not implemented"
)
def
__call__
(
self
,
root
,
**
params
):
self
.
parent_stack
=
[]
self
.
attr_stack
=
[]
for
key
,
value
in
params
.
iteritems
():
setattr
(
self
,
key
,
value
)
root
=
self
.
process_node
(
root
)
for
key
,
value
in
params
.
iteritems
():
delattr
(
self
,
key
)
del
self
.
parent_stack
del
self
.
attr_stack
return
root
class
VisitorTransform
(
Transform
):
# Note: If needed, this can be replaced with a more efficient metaclass
# approach, resolving the jump table at module load time.
def
__init__
(
self
,
readonly
=
False
,
**
kw
):
"""readonly - If this is set to True, the results of process_node
will be discarded (so that one can return None without changing
the tree)."""
super
(
VisitorTransform
,
self
).
__init__
(
**
kw
)
self
.
visitmethods
=
{
'process_'
:
{},
'pre_'
:
{},
'post_'
:
{}}
self
.
attrname
=
""
self
.
readonly
=
readonly
def
get_visitfunc
(
self
,
prefix
,
cls
):
mname
=
prefix
+
cls
.
__name__
m
=
self
.
visitmethods
[
prefix
].
get
(
mname
)
if
m
is
None
:
# Must resolve, try entire hierarchy
for
cls
in
inspect
.
getmro
(
cls
):
m
=
getattr
(
self
,
prefix
+
cls
.
__name__
,
None
)
if
m
is
not
None
:
break
if
m
is
None
:
raise
RuntimeError
(
"Not a Node descendant: "
+
cls
.
__name__
)
self
.
visitmethods
[
prefix
][
mname
]
=
m
return
m
def
process_node
(
self
,
node
,
name
=
"_"
):
# Pass on to calls registered in self.visitmethods
self
.
attrname
=
name
if
node
is
None
:
return
None
result
=
self
.
get_visitfunc
(
"process_"
,
node
.
__class__
)(
node
)
if
self
.
readonly
:
return
node
else
:
return
result
def
process_Node
(
self
,
node
):
descend
=
self
.
get_visitfunc
(
"pre_"
,
node
.
__class__
)(
node
)
if
descend
:
self
.
process_children
(
node
)
self
.
get_visitfunc
(
"post_"
,
node
.
__class__
)(
node
)
return
node
def
pre_Node
(
self
,
node
):
return
True
def
post_Node
(
self
,
node
):
pass
# Utils
def
ensure_statlist
(
node
):
if
not
isinstance
(
node
,
Nodes
.
StatListNode
):
node
=
Nodes
.
StatListNode
(
pos
=
node
.
pos
,
stats
=
[
node
])
return
node
class
PrintTree
(
Transform
):
"""Prints a representation of the tree to standard output.
Subclass and override repr_of to provide more information
...
...
@@ -72,15 +151,24 @@ class PrintTree(Transform):
def
unindent
(
self
):
self
.
_indent
=
self
.
_indent
[:
-
2
]
def
initialize
(
self
,
phase
,
**
option
s
):
def
__call__
(
self
,
tree
,
phase
=
None
,
**
param
s
):
print
(
"Parse tree dump at phase '%s'"
%
phase
)
super
(
PrintTree
,
self
).
__call__
(
tree
,
phase
=
phase
,
**
params
)
# Don't do anything about process_list, the defaults gives
# nice-looking name[idx] nodes which will visually appear
# under the parent-node, not displaying the list itself in
# the hierarchy.
def
process_node
(
self
,
node
,
name
):
def
process_node
(
self
,
node
):
if
len
(
self
.
attr_stack
)
==
0
:
name
=
"(root)"
else
:
attr
,
idx
=
self
.
attr_stack
[
-
1
]
if
idx
is
not
None
:
name
=
"%s[%d]"
%
(
attr
,
idx
)
else
:
name
=
attr
print
(
"%s- %s: %s"
%
(
self
.
_indent
,
name
,
self
.
repr_of
(
node
)))
self
.
indent
()
self
.
process_children
(
node
)
...
...
@@ -92,9 +180,14 @@ class PrintTree(Transform):
return
"(none)"
else
:
result
=
node
.
__class__
.
__name__
if
isinstance
(
node
,
ExprNodes
.
ExprNode
):
if
isinstance
(
node
,
ExprNodes
.
NameNode
):
result
+=
"(type=%s, name=
\
"
%s
\
"
)"
%
(
repr
(
node
.
type
),
node
.
name
)
elif
isinstance
(
node
,
Nodes
.
DefNode
):
result
+=
"(name=
\
"
%s
\
"
)"
%
node
.
name
elif
isinstance
(
node
,
ExprNodes
.
ExprNode
):
t
=
node
.
type
result
+=
"(type=%s)"
%
repr
(
t
)
return
result
...
...
@@ -108,9 +201,8 @@ class TransformSet(dict):
for
name
in
PHASES
:
self
[
name
]
=
[]
def
run
(
self
,
name
,
node
,
**
options
):
assert
name
in
self
assert
name
in
self
,
"Transform phase %s not defined"
%
name
for
transform
in
self
[
name
]:
transform
.
initialize
(
phase
=
name
,
**
options
)
transform
.
process_node
(
node
,
"(root)"
)
transform
(
node
,
phase
=
name
,
**
options
)
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