Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
P
Pyston
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
Boxiang Sun
Pyston
Commits
153fd1eb
Commit
153fd1eb
authored
May 29, 2015
by
Kevin Modzelewski
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #566 from kmod/sampling_prof2
Python-level sampling profiler
parents
d618cd63
cee63186
Changes
13
Hide whitespace changes
Inline
Side-by-side
Showing
13 changed files
with
131 additions
and
8 deletions
+131
-8
src/codegen/ast_interpreter.cpp
src/codegen/ast_interpreter.cpp
+9
-2
src/codegen/codegen.h
src/codegen/codegen.h
+2
-0
src/codegen/entry.cpp
src/codegen/entry.cpp
+13
-0
src/codegen/irgen.cpp
src/codegen/irgen.cpp
+2
-2
src/codegen/irgen/irgenerator.cpp
src/codegen/irgen/irgenerator.cpp
+16
-1
src/codegen/irgen/irgenerator.h
src/codegen/irgen/irgenerator.h
+1
-1
src/codegen/parser.cpp
src/codegen/parser.cpp
+1
-1
src/codegen/unwinding.cpp
src/codegen/unwinding.cpp
+1
-1
src/core/options.h
src/core/options.h
+2
-0
src/core/threading.cpp
src/core/threading.cpp
+14
-0
src/runtime/stacktrace.cpp
src/runtime/stacktrace.cpp
+9
-0
src/runtime/traceback.cpp
src/runtime/traceback.cpp
+1
-0
tools/process_stackdump.py
tools/process_stackdump.py
+60
-0
No files found.
src/codegen/ast_interpreter.cpp
View file @
153fd1eb
...
...
@@ -300,8 +300,6 @@ public:
};
Value
ASTInterpreter
::
execute
(
ASTInterpreter
&
interpreter
,
CFGBlock
*
start_block
,
AST_stmt
*
start_at
)
{
threading
::
allowGLReadPreemption
();
STAT_TIMER
(
t0
,
"us_timer_astinterpreter_execute"
);
void
*
frame_addr
=
__builtin_frame_address
(
0
);
...
...
@@ -315,6 +313,11 @@ Value ASTInterpreter::execute(ASTInterpreter& interpreter, CFGBlock* start_block
start_at
=
start_block
->
body
[
0
];
}
// Important that this happens after RegisterHelper:
interpreter
.
current_inst
=
start_at
;
threading
::
allowGLReadPreemption
();
interpreter
.
current_inst
=
NULL
;
interpreter
.
current_block
=
start_block
;
bool
started
=
false
;
for
(
auto
s
:
start_block
->
body
)
{
...
...
@@ -692,6 +695,10 @@ Value ASTInterpreter::visit_yield(AST_Yield* node) {
}
Value
ASTInterpreter
::
visit_stmt
(
AST_stmt
*
node
)
{
#if ENABLE_SAMPLING_PROFILER
threading
::
allowGLReadPreemption
();
#endif
if
(
0
)
{
printf
(
"%20s % 2d "
,
source_info
->
getName
().
c_str
(),
current_block
->
idx
);
print_ast
(
node
);
...
...
src/codegen/codegen.h
View file @
153fd1eb
...
...
@@ -87,6 +87,8 @@ extern GlobalState g;
// in runtime_hooks.cpp:
void
initGlobalFuncs
(
GlobalState
&
g
);
extern
int
sigprof_pending
;
DS_DECLARE_RWLOCK
(
codegen_rwlock
);
}
...
...
src/codegen/entry.cpp
View file @
153fd1eb
...
...
@@ -55,6 +55,8 @@
namespace
pyston
{
int
sigprof_pending
=
0
;
GlobalState
g
;
extern
"C"
{
...
...
@@ -356,6 +358,10 @@ static void handle_sigusr1(int signum) {
_printStacktrace
();
}
static
void
handle_sigprof
(
int
signum
)
{
sigprof_pending
++
;
}
static
void
handle_sigint
(
int
signum
)
{
assert
(
signum
==
SIGINT
);
// TODO: this should set a flag saying a KeyboardInterrupt is pending.
...
...
@@ -461,6 +467,13 @@ void initCodegen() {
signal
(
SIGUSR1
,
&
handle_sigusr1
);
signal
(
SIGINT
,
&
handle_sigint
);
#if ENABLE_SAMPLING_PROFILER
struct
itimerval
prof_timer
;
prof_timer
.
it_value
.
tv_sec
=
prof_timer
.
it_interval
.
tv_sec
=
0
;
prof_timer
.
it_value
.
tv_usec
=
prof_timer
.
it_interval
.
tv_usec
=
1000
;
signal
(
SIGPROF
,
handle_sigprof
);
setitimer
(
ITIMER_PROF
,
&
prof_timer
,
NULL
);
#endif
// There are some parts of llvm that are only configurable through command line args,
// so construct a fake argc/argv pair and pass it to the llvm command line machinery:
...
...
src/codegen/irgen.cpp
View file @
153fd1eb
...
...
@@ -594,7 +594,7 @@ static void emitBBs(IRGenState* irstate, TypeAnalysis* types, const OSREntryDesc
// Function-entry safepoint:
// TODO might be more efficient to do post-call safepoints?
generator
->
doSafePoint
();
generator
->
doSafePoint
(
block
->
body
[
0
]
);
}
else
if
(
entry_descriptor
&&
block
==
entry_descriptor
->
backedge
->
target
)
{
assert
(
block
->
predecessors
.
size
()
>
1
);
assert
(
osr_entry_block
);
...
...
@@ -763,7 +763,7 @@ static void emitBBs(IRGenState* irstate, TypeAnalysis* types, const OSREntryDesc
if
(
predecessor
->
idx
>
block
->
idx
)
{
// Loop safepoint:
// TODO does it matter which side of the backedge these are on?
generator
->
doSafePoint
();
generator
->
doSafePoint
(
block
->
body
[
0
]
);
break
;
}
}
...
...
src/codegen/irgen/irgenerator.cpp
View file @
153fd1eb
...
...
@@ -2608,6 +2608,13 @@ public:
if
(
state
==
DEAD
)
break
;
assert
(
state
!=
FINISHED
);
#if ENABLE_SAMPLING_PROFILER
auto
stmt
=
block
->
body
[
i
];
if
(
stmt
->
type
!=
AST_TYPE
::
Assign
)
// could be a landingpad
doSafePoint
(
block
->
body
[
i
]);
#endif
doStmt
(
block
->
body
[
i
],
UnwindInfo
(
block
->
body
[
i
],
NULL
));
}
if
(
VERBOSITY
(
"irgenerator"
)
>=
2
)
{
// print ending symbol table
...
...
@@ -2618,7 +2625,15 @@ public:
}
}
void
doSafePoint
()
override
{
emitter
.
getBuilder
()
->
CreateCall
(
g
.
funcs
.
allowGLReadPreemption
);
}
void
doSafePoint
(
AST_stmt
*
next_statement
)
override
{
// If the sampling profiler is turned on (and eventually, destructors), we need frame-introspection
// support while in allowGLReadPreemption:
#if ENABLE_SAMPLING_PROFILER
emitter
.
createCall
(
UnwindInfo
(
next_statement
,
NULL
),
g
.
funcs
.
allowGLReadPreemption
);
#else
emitter
.
getBuilder
()
->
CreateCall
(
g
.
funcs
.
allowGLReadPreemption
);
#endif
}
};
IRGenerator
*
createIRGenerator
(
IRGenState
*
irstate
,
std
::
unordered_map
<
CFGBlock
*
,
llvm
::
BasicBlock
*>&
entry_blocks
,
...
...
src/codegen/irgen/irgenerator.h
View file @
153fd1eb
...
...
@@ -127,7 +127,7 @@ public:
virtual
void
copySymbolsFrom
(
SymbolTable
*
st
)
=
0
;
virtual
void
run
(
const
CFGBlock
*
block
)
=
0
;
// primary entry point
virtual
EndingState
getEndingSymbolTable
()
=
0
;
virtual
void
doSafePoint
()
=
0
;
virtual
void
doSafePoint
(
AST_stmt
*
next_statement
)
=
0
;
virtual
void
addFrameStackmapArgs
(
PatchpointInfo
*
pp
,
AST_stmt
*
current_stmt
,
std
::
vector
<
llvm
::
Value
*>&
stackmap_args
)
=
0
;
};
...
...
src/codegen/parser.cpp
View file @
153fd1eb
...
...
@@ -1000,7 +1000,7 @@ AST_Module* parse_file(const char* fn) {
if
(
ENABLE_PYPA_PARSER
)
{
AST_Module
*
rtn
=
pypa_parse
(
fn
);
RELEASE_ASSERT
(
rtn
,
"unknown parse error
"
);
RELEASE_ASSERT
(
rtn
,
"unknown parse error
(possibly: '%s'?)"
,
strerror
(
errno
)
);
return
rtn
;
}
...
...
src/codegen/unwinding.cpp
View file @
153fd1eb
...
...
@@ -345,7 +345,7 @@ public:
return
reinterpret_cast
<
AST_stmt
*>
(
readLocation
(
e
.
locations
[
0
]));
}
}
abort
(
);
RELEASE_ASSERT
(
0
,
"no frame info found at offset 0x%x / ip 0x%lx!"
,
offset
,
ip
);
}
else
if
(
id
.
type
==
PythonFrameId
::
INTERPRETED
)
{
return
getCurrentStatementForInterpretedFrame
((
void
*
)
id
.
bp
);
}
...
...
src/core/options.h
View file @
153fd1eb
...
...
@@ -47,6 +47,8 @@ extern bool ENABLE_ICS, ENABLE_ICGENERICS, ENABLE_ICGETITEMS, ENABLE_ICSETITEMS,
// Due to a temporary LLVM limitation, represent bools as i64's instead of i1's.
extern
bool
BOOLS_AS_I64
;
#define ENABLE_SAMPLING_PROFILER 0
}
}
...
...
src/core/threading.cpp
View file @
153fd1eb
...
...
@@ -24,12 +24,14 @@
#include "Python.h"
#include "codegen/codegen.h" // sigprof_pending
#include "core/common.h"
#include "core/options.h"
#include "core/stats.h"
#include "core/thread_utils.h"
#include "core/util.h"
#include "gc/collector.h"
#include "runtime/objmodel.h" // _printStacktrace
namespace
pyston
{
namespace
threading
{
...
...
@@ -528,6 +530,18 @@ int gil_check_count = 0;
// We could enforce fairness by having a FIFO of events (implementd with mutexes?)
// and make sure to always wake up the longest-waiting one.
void
allowGLReadPreemption
()
{
#if ENABLE_SAMPLING_PROFILER
if
(
unlikely
(
sigprof_pending
))
{
// Output multiple stacktraces if we received multiple signals
// between being able to handle it (such as being in LLVM or the GC),
// to try to fully account for that time.
while
(
sigprof_pending
)
{
_printStacktrace
();
sigprof_pending
--
;
}
}
#endif
// Double-checked locking: first read with no ordering constraint:
if
(
!
threads_waiting_on_gil
.
load
(
std
::
memory_order_relaxed
))
return
;
...
...
src/runtime/stacktrace.cpp
View file @
153fd1eb
...
...
@@ -101,7 +101,16 @@ void raiseSyntaxErrorHelper(const std::string& file, const std::string& func, AS
}
void
_printStacktrace
()
{
static
bool
recursive
=
false
;
if
(
recursive
)
{
fprintf
(
stderr
,
"_printStacktrace ran into an issue; refusing to try it again!
\n
"
);
return
;
}
recursive
=
true
;
printTraceback
(
getTraceback
());
recursive
=
false
;
}
// where should this go...
...
...
src/runtime/traceback.cpp
View file @
153fd1eb
...
...
@@ -85,6 +85,7 @@ void printTraceback(Box* b) {
fprintf
(
stderr
,
" %.*s
\n
"
,
(
int
)
r
,
ptr
);
free
(
buf
);
}
fclose
(
f
);
}
}
}
...
...
tools/process_stackdump.py
0 → 100644
View file @
153fd1eb
import
sys
if
__name__
==
"__main__"
:
fn
=
sys
.
argv
[
1
]
tracebacks
=
[]
cur_traceback
=
[]
with
open
(
fn
)
as
f
:
for
l
in
f
:
if
l
.
startswith
(
"Traceback"
):
if
cur_traceback
:
tracebacks
.
append
(
''
.
join
(
cur_traceback
).
rstrip
())
cur_traceback
=
[]
elif
not
(
l
.
startswith
(
" File"
)
or
l
.
startswith
(
" "
)):
print
"non-traceback line? "
,
l
.
strip
()
continue
else
:
cur_traceback
.
append
(
l
)
if
cur_traceback
:
tracebacks
.
append
(
''
.
join
(
cur_traceback
).
rstrip
())
cur_traceback
=
[]
counts
=
{}
for
t
in
tracebacks
:
locations
=
[
l
for
l
in
t
.
split
(
'
\
n
'
)
if
l
.
startswith
(
" File"
)]
last_file
=
locations
[
-
1
].
split
(
'"'
)[
1
]
last_function
=
locations
[
-
1
].
split
()[
-
1
][:
-
1
]
# dedupe on:
# key = t # full traceback
# key = '\n'.join(t.split('\n')[-8:]) # last 4 stack frames
# key = '\n'.join(t.split('\n')[-4:]) # last 2 stack frames
# key = '\n'.join(t.split('\n')[-2:]) # last stack frame
# key = last_file, last_function
key
=
last_file
counts
[
key
]
=
counts
.
get
(
key
,
0
)
+
1
n
=
len
(
tracebacks
)
NUM_DISPLAY
=
6
entries
=
sorted
(
counts
.
items
(),
key
=
lambda
(
k
,
v
):
v
)
if
len
(
counts
)
>
NUM_DISPLAY
:
num_hidden
=
0
counts_hidden
=
0
for
k
,
v
in
entries
[:
-
NUM_DISPLAY
]:
num_hidden
+=
1
counts_hidden
+=
v
print
"Hiding %d entries that occurred %d (%.1f%%) times"
%
(
num_hidden
,
counts_hidden
,
100.0
*
counts_hidden
/
n
)
for
k
,
v
in
sorted
(
counts
.
items
(),
key
=
lambda
(
k
,
v
):
v
)[
-
NUM_DISPLAY
:]:
print
print
"Occurs %d (%.1f%%) times:"
%
(
v
,
100.0
*
v
/
n
)
print
k
print
print
"Total tracebacks:"
,
n
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