This repository has been archived by the owner on Sep 11, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 27
/
profile.cpp
156 lines (133 loc) · 4.35 KB
/
profile.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
/* SPDX-License-Identifier: GPL-3.0-or-later */
/* Copyright © 2016-2023 Byteduck */
#include <libdebug/LiveDebugger.h>
#include <csignal>
#include <memory>
#include <sys/wait.h>
#include <libduck/FormatStream.h>
#include <libduck/Args.h>
#include <libduck/Socket.h>
#include <ctime>
constexpr int debugd_port = 59336;
constexpr const char* debugd_start = "DEBUGD\nPROFILE\n";
using namespace Debug;
using Duck::OutputStream;
int pid;
int interval = 10;
int duration = 5000;
bool remote = false;
std::string filename;
struct Thread {
tid_t tid;
LiveDebugger debugger;
std::vector<std::vector<size_t>> stacks;
};
Duck::Result output_profile(
Duck::OutputStream& stream,
std::map<size_t, AddressInfo>& symbols,
std::vector<Duck::Ptr<Thread>>& threads) {
for (auto& thread : threads){
for (auto& stk : thread->stacks) {
stream << "thread " << thread->tid << ";";
for(size_t i = stk.size(); i > 0; i--) {
auto pos = stk[i - 1];
auto info = symbols.find(pos);
if (info == symbols.end()) {
auto info_res = thread->debugger.info_at(pos);
if(info_res.is_error())
symbols[pos] = {"???", pos, nullptr};
else
symbols[pos] = info_res.value();
info = symbols.find(pos);
}
auto& symbol = info->second;
if (!symbol.object)
stream % "?? @ {#x}" % symbol.symbol_offset;
else if(symbol.symbol_name == "__syscall_trap__")
stream << "<kernel>";
else
stream % "{} @ {}" % symbol.symbol_name % symbol.object->name;
if (i != 1)
stream << ";";
}
stream << " 1\n";
}
}
return Duck::Result::SUCCESS;
}
Duck::Result profile() {
auto proc = TRY(Sys::Process::get(pid));
// First, attach to all the threads
const int loops = duration / interval;
std::vector<Duck::Ptr<Thread>> threads;
for (auto tid : proc.threads()) {
auto thread = std::make_shared<Thread>();
thread->tid = tid;
thread->stacks.reserve(loops);
auto attach_res = thread->debugger.attach(pid, tid);
if (attach_res.is_error())
Duck::printerrln("Warning: Failed to attach to thread {}: {}", tid, attach_res);
else
threads.push_back(thread);
}
if (threads.empty())
return Duck::Result("Could not attach to any threads");
Duck::println("Sampling {} for ~{}ms...", proc.name(), duration);
for (size_t i = 0; i < loops; i++) {
waitpid(pid, nullptr, 0);
for (auto& thread : threads)
thread->stacks.push_back(TRY(thread->debugger.walk_stack_unsymbolicated()));
kill(pid, SIGCONT);
usleep(interval * 1000);
kill(pid, SIGSTOP);
}
waitpid(pid, nullptr, 0);
kill(pid, SIGCONT);
Duck::println("Done! Symbolicating and dumping...");
std::map<size_t, AddressInfo> symbols;
if (remote) {
// Collect into StringOutputStream
Duck::StringOutputStream stream;
TRYRES(output_profile(stream, symbols, threads));
// Connect to socket
Duck::print("Connecting to debug daemon... ", debugd_port);
fflush(stdout);
auto sock = TRY(Duck::Socket::open(Duck::Socket::Inet, Duck::Socket::Stream, Duck::Socket::TCP));
TRYRES(sock.connect({10, 0, 2, 2}, debugd_port));
// Write to socket
Duck::print("Sending... ");
fflush(stdout);
TRYRES(sock.send(debugd_start, strlen(debugd_start)));
for (size_t i = 0; i < stream.string().length() + 1; i += 1024) {
auto len = std::min((size_t) 1024, stream.string().length() + 1 - i);
TRYRES(sock.send(stream.string().c_str() + i, len));
}
Duck::println("Sent!");
sock.close();
} else {
// Write to file
if (filename.empty())
filename = "profile-" + proc.name() + "-" + std::to_string(std::time(nullptr)) + ".txt";
auto out = TRY(Duck::File::open(filename, "w"));
Duck::FileOutputStream fs {out};
TRYRES(output_profile(fs, symbols, threads));
out.close();
Duck::println("Done! Saved to {}.", filename);
}
return Duck::Result::SUCCESS;
}
int main(int argc, char** argv) {
Duck::Args args;
args.add_positional(pid, true, "pid", "The PID of the program to profile.");
args.add_named(interval, "i", "interval", "The interval with which to sample, in ms. (Default: 10)");
args.add_named(duration, "d", "duration", "The duration to sample for, in ms. (Default: 5000)");
args.add_named(filename, "o", "output", "The output file.");
args.add_flag(remote, "r", "remote", "Sends the output to a remote debug server.");
args.parse(argc, argv);
auto res = profile();
if (res.is_error()) {
printf("Error: %s\n", res.message().c_str());
return res.code();
}
return 0;
}