Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cpython
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
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
cpython
Commits
355f3e1e
Commit
355f3e1e
authored
Sep 12, 2019
by
Mark Shannon
Committed by
T. Wouters
Sep 12, 2019
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix depth-first-search computation in compile.c (GH-16042)
parent
8c74574e
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
24 additions
and
30 deletions
+24
-30
Python/compile.c
Python/compile.c
+24
-30
No files found.
Python/compile.c
View file @
355f3e1e
...
@@ -5373,7 +5373,7 @@ compiler_visit_slice(struct compiler *c, slice_ty s, expr_context_ty ctx)
...
@@ -5373,7 +5373,7 @@ compiler_visit_slice(struct compiler *c, slice_ty s, expr_context_ty ctx)
/* End of the compiler section, beginning of the assembler section */
/* End of the compiler section, beginning of the assembler section */
/* do depth-first search of basic block graph, starting with block.
/* do depth-first search of basic block graph, starting with block.
post records the block indices in post-
order.
a_order records the block indices in
order.
XXX must handle implicit jumps from one block to next
XXX must handle implicit jumps from one block to next
*/
*/
...
@@ -5382,7 +5382,7 @@ struct assembler {
...
@@ -5382,7 +5382,7 @@ struct assembler {
PyObject
*
a_bytecode
;
/* string containing bytecode */
PyObject
*
a_bytecode
;
/* string containing bytecode */
int
a_offset
;
/* offset into bytecode */
int
a_offset
;
/* offset into bytecode */
int
a_nblocks
;
/* number of reachable blocks */
int
a_nblocks
;
/* number of reachable blocks */
basicblock
**
a_
postorder
;
/* list of blocks in dfs post
order */
basicblock
**
a_
order
;
/* list of blocks in dfs
order */
PyObject
*
a_lnotab
;
/* string containing lnotab */
PyObject
*
a_lnotab
;
/* string containing lnotab */
int
a_lnotab_off
;
/* offset into lnotab */
int
a_lnotab_off
;
/* offset into lnotab */
int
a_lineno
;
/* last lineno of emitted instruction */
int
a_lineno
;
/* last lineno of emitted instruction */
...
@@ -5390,28 +5390,21 @@ struct assembler {
...
@@ -5390,28 +5390,21 @@ struct assembler {
};
};
static
void
static
void
dfs
(
struct
compiler
*
c
,
basicblock
*
b
,
struct
assembler
*
a
,
int
end
)
dfs
(
struct
compiler
*
c
,
basicblock
*
entry
,
struct
assembler
*
a
)
{
{
int
i
,
j
;
/* Avoid excessive recursion by following 'next' links to the
* end of the chain before handling any branches */
/* Get rid of recursion for normal control flow.
for
(
basicblock
*
b
=
entry
;
b
;
b
=
b
->
b_next
)
{
Since the number of blocks is limited, unused space in a_postorder
(from a_nblocks to end) can be used as a stack for still not ordered
blocks. */
for
(
j
=
end
;
b
&&
!
b
->
b_seen
;
b
=
b
->
b_next
)
{
b
->
b_seen
=
1
;
b
->
b_seen
=
1
;
assert
(
a
->
a_nblocks
<
j
);
a
->
a_order
[
a
->
a_nblocks
++
]
=
b
;
a
->
a_postorder
[
--
j
]
=
b
;
}
for
(
basicblock
*
b
=
entry
;
b
;
b
=
b
->
b_next
)
{
for
(
int
i
=
0
;
i
<
b
->
b_iused
;
i
++
)
{
basicblock
*
target
=
b
->
b_instr
[
i
].
i_target
;
if
(
target
&&
!
target
->
b_seen
)
{
dfs
(
c
,
target
,
a
);
}
}
while
(
j
<
end
)
{
b
=
a
->
a_postorder
[
j
++
];
for
(
i
=
0
;
i
<
b
->
b_iused
;
i
++
)
{
struct
instr
*
instr
=
&
b
->
b_instr
[
i
];
if
(
instr
->
i_jrel
||
instr
->
i_jabs
)
dfs
(
c
,
instr
->
i_target
,
a
,
j
);
}
}
assert
(
a
->
a_nblocks
<
j
);
a
->
a_postorder
[
a
->
a_nblocks
++
]
=
b
;
}
}
}
}
...
@@ -5517,9 +5510,9 @@ assemble_init(struct assembler *a, int nblocks, int firstlineno)
...
@@ -5517,9 +5510,9 @@ assemble_init(struct assembler *a, int nblocks, int firstlineno)
PyErr_NoMemory
();
PyErr_NoMemory
();
return
0
;
return
0
;
}
}
a
->
a_
post
order
=
(
basicblock
**
)
PyObject_Malloc
(
a
->
a_order
=
(
basicblock
**
)
PyObject_Malloc
(
sizeof
(
basicblock
*
)
*
nblocks
);
sizeof
(
basicblock
*
)
*
nblocks
);
if
(
!
a
->
a_
post
order
)
{
if
(
!
a
->
a_order
)
{
PyErr_NoMemory
();
PyErr_NoMemory
();
return
0
;
return
0
;
}
}
...
@@ -5531,8 +5524,8 @@ assemble_free(struct assembler *a)
...
@@ -5531,8 +5524,8 @@ assemble_free(struct assembler *a)
{
{
Py_XDECREF
(
a
->
a_bytecode
);
Py_XDECREF
(
a
->
a_bytecode
);
Py_XDECREF
(
a
->
a_lnotab
);
Py_XDECREF
(
a
->
a_lnotab
);
if
(
a
->
a_
post
order
)
if
(
a
->
a_order
)
PyObject_Free
(
a
->
a_
post
order
);
PyObject_Free
(
a
->
a_order
);
}
}
static
int
static
int
...
@@ -5693,8 +5686,8 @@ assemble_jump_offsets(struct assembler *a, struct compiler *c)
...
@@ -5693,8 +5686,8 @@ assemble_jump_offsets(struct assembler *a, struct compiler *c)
Replace block pointer with position in bytecode. */
Replace block pointer with position in bytecode. */
do
{
do
{
totsize
=
0
;
totsize
=
0
;
for
(
i
=
a
->
a_nblocks
-
1
;
i
>=
0
;
i
--
)
{
for
(
i
=
0
;
i
<
a
->
a_nblocks
;
i
++
)
{
b
=
a
->
a_
post
order
[
i
];
b
=
a
->
a_order
[
i
];
bsize
=
blocksize
(
b
);
bsize
=
blocksize
(
b
);
b
->
b_offset
=
totsize
;
b
->
b_offset
=
totsize
;
totsize
+=
bsize
;
totsize
+=
bsize
;
...
@@ -5998,14 +5991,15 @@ assemble(struct compiler *c, int addNone)
...
@@ -5998,14 +5991,15 @@ assemble(struct compiler *c, int addNone)
}
}
if
(
!
assemble_init
(
&
a
,
nblocks
,
c
->
u
->
u_firstlineno
))
if
(
!
assemble_init
(
&
a
,
nblocks
,
c
->
u
->
u_firstlineno
))
goto
error
;
goto
error
;
dfs
(
c
,
entryblock
,
&
a
,
nblocks
);
dfs
(
c
,
entryblock
,
&
a
);
assert
(
a
.
a_nblocks
<=
nblocks
);
/* Can't modify the bytecode after computing jump offsets. */
/* Can't modify the bytecode after computing jump offsets. */
assemble_jump_offsets
(
&
a
,
c
);
assemble_jump_offsets
(
&
a
,
c
);
/* Emit code in
reverse post
order from dfs. */
/* Emit code in order from dfs. */
for
(
i
=
a
.
a_nblocks
-
1
;
i
>=
0
;
i
--
)
{
for
(
i
=
0
;
i
<
a
.
a_nblocks
;
i
++
)
{
b
=
a
.
a_
post
order
[
i
];
b
=
a
.
a_order
[
i
];
for
(
j
=
0
;
j
<
b
->
b_iused
;
j
++
)
for
(
j
=
0
;
j
<
b
->
b_iused
;
j
++
)
if
(
!
assemble_emit
(
&
a
,
&
b
->
b_instr
[
j
]))
if
(
!
assemble_emit
(
&
a
,
&
b
->
b_instr
[
j
]))
goto
error
;
goto
error
;
...
...
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