Skip to content

Commit 70ca5ca

Browse files
committed
Allowed tracing a single Python thread
This is useful for running PyFlame from Jupyter Notebook, where a cell could be isolated to a separate thread prior to profiling. See also uber-archive#163
1 parent d67c353 commit 70ca5ca

File tree

3 files changed

+41
-2
lines changed

3 files changed

+41
-2
lines changed

src/prober.cc

+14-2
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@ static const char usage_str[] =
4848
"Common Options:\n"
4949
#ifdef ENABLE_THREADS
5050
" --threads Enable multi-threading support\n"
51+
" --only=IDENT Only trace a thread identified by IDENT (requires --threads)\n"
5152
" -d, --dump Dump stacks from all threads (implies --threads)\n"
5253
#else
5354
" -d, --dump Dump the current interpreter stack\n"
@@ -189,6 +190,7 @@ int Prober::ParseOpts(int argc, char **argv) {
189190
{"seconds", required_argument, 0, 's'},
190191
#if ENABLE_THREADS
191192
{"threads", no_argument, 0, 'L'},
193+
{"only", required_argument, 0, 'i'},
192194
#endif
193195
{"no-line-numbers", no_argument, 0, 'n'},
194196
{"output", required_argument, 0, 'o'},
@@ -238,7 +240,10 @@ int Prober::ParseOpts(int argc, char **argv) {
238240
std::cout << PYFLAME_VERSION_STR << "\n\n" << usage_str;
239241
return 0;
240242
break;
241-
#ifdef ENABLE_THREADS
243+
#if ENABLE_THREADS
244+
case 'i':
245+
thread_id_ = std::stoul(optarg);
246+
break;
242247
case 'L':
243248
enable_threads_ = true;
244249
break;
@@ -309,6 +314,11 @@ int Prober::ParseOpts(int argc, char **argv) {
309314
std::cerr << "WARNING: Specifying a PID to trace without -p is deprecated; "
310315
"see Pyflame issue #99 for details.\n";
311316
}
317+
if (thread_id_ > 0 && !enable_threads_) {
318+
std::cerr << "Option --only requires --threads.\n";
319+
std::cerr << usage_str;
320+
return 1;
321+
}
312322
interval_ = ToMicroseconds(sample_rate_);
313323
return -1;
314324
}
@@ -416,7 +426,9 @@ int Prober::ProbeLoop(const PyFrob &frobber, std::ostream *out) {
416426
}
417427

418428
for (const auto &thread : threads) {
419-
call_stacks.push_back({now, thread.frames()});
429+
if (thread_id_ == 0 || thread.id() == thread_id_) {
430+
call_stacks.push_back({now, thread.frames()});
431+
}
420432
}
421433

422434
if (check_end && (now + interval_ >= end)) {

src/prober.h

+2
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ class Prober {
3939
include_ts_(false),
4040
include_line_number_(true),
4141
enable_threads_(false),
42+
thread_id_(0),
4243
seconds_(1),
4344
sample_rate_(0.01) {}
4445
Prober(const Prober &other) = delete;
@@ -63,6 +64,7 @@ class Prober {
6364
bool include_ts_;
6465
bool include_line_number_;
6566
bool enable_threads_;
67+
unsigned long thread_id_;
6668
double seconds_;
6769
double sample_rate_;
6870
std::chrono::microseconds interval_;

tests/test_end_to_end.py

+25
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
import subprocess
2323
import sys
2424
import time
25+
import threading
2526

2627
IDLE_RE = re.compile(r'^\(idle\) \d+$')
2728
FLAMEGRAPH_RE = re.compile(
@@ -637,3 +638,27 @@ def test_no_line_numbers(dijkstra):
637638
for line in lines:
638639
assert_flamegraph(
639640
line, allow_idle=True, line_re=FLAMEGRAPH_NONUMBER_RE)
641+
642+
643+
def test_only():
644+
should_stop = False
645+
646+
def infinite_sleeper():
647+
while not should_stop:
648+
time.sleep(0.01)
649+
650+
thread = threading.Thread(target=infinite_sleeper)
651+
thread.daemon = True
652+
thread.start()
653+
654+
proc = subprocess.Popen(
655+
[path_to_pyflame(), '-p',
656+
str(os.getpid()), '--seconds=1',
657+
'--threads', '--only={}'.format(thread.ident)],
658+
stdout=subprocess.PIPE,
659+
stderr=subprocess.PIPE,
660+
universal_newlines=True)
661+
out, err = communicate(proc)
662+
assert not err
663+
assert proc.returncode == 0
664+
assert 'test_only' not in out

0 commit comments

Comments
 (0)