提交 26f27380 编写于 作者: I Ingo Molnar

Merge tag 'perf-core-for-mingo' of...

Merge tag 'perf-core-for-mingo' of git://git.kernel.org/pub/scm/linux/kernel/git/jolsa/perf into perf/core

Pull perf/core improvements and fixes from Jiri Olsa:

  * Propagate exit status of a command line workload for
    record command (Namhyung Kim)

  * Use tid for finding thread (Namhyung Kim)

  * Clarify the output of perf sched map plus small sched
    command fixies (Dongsheng Yang)
Signed-off-by: NJiri Olsa <jolsa@kernel.org>
Signed-off-by: NIngo Molnar <mingo@kernel.org>
...@@ -209,7 +209,7 @@ static int perf_event__inject_buildid(struct perf_tool *tool, ...@@ -209,7 +209,7 @@ static int perf_event__inject_buildid(struct perf_tool *tool,
cpumode = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK; cpumode = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;
thread = machine__findnew_thread(machine, sample->pid, sample->pid); thread = machine__findnew_thread(machine, sample->pid, sample->tid);
if (thread == NULL) { if (thread == NULL) {
pr_err("problem processing %d event, skipping it.\n", pr_err("problem processing %d event, skipping it.\n",
event->header.type); event->header.type);
......
...@@ -235,7 +235,7 @@ static int process_sample_event(struct perf_tool *tool __maybe_unused, ...@@ -235,7 +235,7 @@ static int process_sample_event(struct perf_tool *tool __maybe_unused,
struct machine *machine) struct machine *machine)
{ {
struct thread *thread = machine__findnew_thread(machine, sample->pid, struct thread *thread = machine__findnew_thread(machine, sample->pid,
sample->pid); sample->tid);
if (thread == NULL) { if (thread == NULL) {
pr_debug("problem processing %d event, skipping it.\n", pr_debug("problem processing %d event, skipping it.\n",
......
...@@ -30,37 +30,6 @@ ...@@ -30,37 +30,6 @@
#include <sched.h> #include <sched.h>
#include <sys/mman.h> #include <sys/mman.h>
#ifndef HAVE_ON_EXIT_SUPPORT
#ifndef ATEXIT_MAX
#define ATEXIT_MAX 32
#endif
static int __on_exit_count = 0;
typedef void (*on_exit_func_t) (int, void *);
static on_exit_func_t __on_exit_funcs[ATEXIT_MAX];
static void *__on_exit_args[ATEXIT_MAX];
static int __exitcode = 0;
static void __handle_on_exit_funcs(void);
static int on_exit(on_exit_func_t function, void *arg);
#define exit(x) (exit)(__exitcode = (x))
static int on_exit(on_exit_func_t function, void *arg)
{
if (__on_exit_count == ATEXIT_MAX)
return -ENOMEM;
else if (__on_exit_count == 0)
atexit(__handle_on_exit_funcs);
__on_exit_funcs[__on_exit_count] = function;
__on_exit_args[__on_exit_count++] = arg;
return 0;
}
static void __handle_on_exit_funcs(void)
{
int i;
for (i = 0; i < __on_exit_count; i++)
__on_exit_funcs[i] (__exitcode, __on_exit_args[i]);
}
#endif
struct record { struct record {
struct perf_tool tool; struct perf_tool tool;
...@@ -147,29 +116,19 @@ static void sig_handler(int sig) ...@@ -147,29 +116,19 @@ static void sig_handler(int sig)
{ {
if (sig == SIGCHLD) if (sig == SIGCHLD)
child_finished = 1; child_finished = 1;
else
signr = sig;
done = 1; done = 1;
signr = sig;
} }
static void record__sig_exit(int exit_status __maybe_unused, void *arg) static void record__sig_exit(void)
{ {
struct record *rec = arg; if (signr == -1)
int status;
if (rec->evlist->workload.pid > 0) {
if (!child_finished)
kill(rec->evlist->workload.pid, SIGTERM);
wait(&status);
if (WIFSIGNALED(status))
psignal(WTERMSIG(status), rec->progname);
}
if (signr == -1 || signr == SIGUSR1)
return; return;
signal(signr, SIG_DFL); signal(signr, SIG_DFL);
raise(signr);
} }
static int record__open(struct record *rec) static int record__open(struct record *rec)
...@@ -243,27 +202,6 @@ static int process_buildids(struct record *rec) ...@@ -243,27 +202,6 @@ static int process_buildids(struct record *rec)
size, &build_id__mark_dso_hit_ops); size, &build_id__mark_dso_hit_ops);
} }
static void record__exit(int status, void *arg)
{
struct record *rec = arg;
struct perf_data_file *file = &rec->file;
if (status != 0)
return;
if (!file->is_pipe) {
rec->session->header.data_size += rec->bytes_written;
if (!rec->no_buildid)
process_buildids(rec);
perf_session__write_header(rec->session, rec->evlist,
file->fd, true);
perf_session__delete(rec->session);
perf_evlist__delete(rec->evlist);
symbol__exit();
}
}
static void perf_event__synthesize_guest_os(struct machine *machine, void *data) static void perf_event__synthesize_guest_os(struct machine *machine, void *data)
{ {
int err; int err;
...@@ -344,18 +282,19 @@ static volatile int workload_exec_errno; ...@@ -344,18 +282,19 @@ static volatile int workload_exec_errno;
* if the fork fails, since we asked by setting its * if the fork fails, since we asked by setting its
* want_signal to true. * want_signal to true.
*/ */
static void workload_exec_failed_signal(int signo, siginfo_t *info, static void workload_exec_failed_signal(int signo __maybe_unused,
siginfo_t *info,
void *ucontext __maybe_unused) void *ucontext __maybe_unused)
{ {
workload_exec_errno = info->si_value.sival_int; workload_exec_errno = info->si_value.sival_int;
done = 1; done = 1;
signr = signo;
child_finished = 1; child_finished = 1;
} }
static int __cmd_record(struct record *rec, int argc, const char **argv) static int __cmd_record(struct record *rec, int argc, const char **argv)
{ {
int err; int err;
int status = 0;
unsigned long waking = 0; unsigned long waking = 0;
const bool forks = argc > 0; const bool forks = argc > 0;
struct machine *machine; struct machine *machine;
...@@ -367,7 +306,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -367,7 +306,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
rec->progname = argv[0]; rec->progname = argv[0];
on_exit(record__sig_exit, rec); atexit(record__sig_exit);
signal(SIGCHLD, sig_handler); signal(SIGCHLD, sig_handler);
signal(SIGINT, sig_handler); signal(SIGINT, sig_handler);
signal(SIGTERM, sig_handler); signal(SIGTERM, sig_handler);
...@@ -388,32 +327,28 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -388,32 +327,28 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
workload_exec_failed_signal); workload_exec_failed_signal);
if (err < 0) { if (err < 0) {
pr_err("Couldn't run the workload!\n"); pr_err("Couldn't run the workload!\n");
status = err;
goto out_delete_session; goto out_delete_session;
} }
} }
if (record__open(rec) != 0) { if (record__open(rec) != 0) {
err = -1; err = -1;
goto out_delete_session; goto out_child;
} }
if (!rec->evlist->nr_groups) if (!rec->evlist->nr_groups)
perf_header__clear_feat(&session->header, HEADER_GROUP_DESC); perf_header__clear_feat(&session->header, HEADER_GROUP_DESC);
/*
* perf_session__delete(session) will be called at record__exit()
*/
on_exit(record__exit, rec);
if (file->is_pipe) { if (file->is_pipe) {
err = perf_header__write_pipe(file->fd); err = perf_header__write_pipe(file->fd);
if (err < 0) if (err < 0)
goto out_delete_session; goto out_child;
} else { } else {
err = perf_session__write_header(session, rec->evlist, err = perf_session__write_header(session, rec->evlist,
file->fd, false); file->fd, false);
if (err < 0) if (err < 0)
goto out_delete_session; goto out_child;
} }
if (!rec->no_buildid if (!rec->no_buildid
...@@ -421,7 +356,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -421,7 +356,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
pr_err("Couldn't generate buildids. " pr_err("Couldn't generate buildids. "
"Use --no-buildid to profile anyway.\n"); "Use --no-buildid to profile anyway.\n");
err = -1; err = -1;
goto out_delete_session; goto out_child;
} }
machine = &session->machines.host; machine = &session->machines.host;
...@@ -431,7 +366,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -431,7 +366,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
process_synthesized_event); process_synthesized_event);
if (err < 0) { if (err < 0) {
pr_err("Couldn't synthesize attrs.\n"); pr_err("Couldn't synthesize attrs.\n");
goto out_delete_session; goto out_child;
} }
if (have_tracepoints(&rec->evlist->entries)) { if (have_tracepoints(&rec->evlist->entries)) {
...@@ -447,7 +382,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -447,7 +382,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
process_synthesized_event); process_synthesized_event);
if (err <= 0) { if (err <= 0) {
pr_err("Couldn't record tracing data.\n"); pr_err("Couldn't record tracing data.\n");
goto out_delete_session; goto out_child;
} }
rec->bytes_written += err; rec->bytes_written += err;
} }
...@@ -475,7 +410,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -475,7 +410,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
err = __machine__synthesize_threads(machine, tool, &opts->target, rec->evlist->threads, err = __machine__synthesize_threads(machine, tool, &opts->target, rec->evlist->threads,
process_synthesized_event, opts->sample_address); process_synthesized_event, opts->sample_address);
if (err != 0) if (err != 0)
goto out_delete_session; goto out_child;
if (rec->realtime_prio) { if (rec->realtime_prio) {
struct sched_param param; struct sched_param param;
...@@ -484,7 +419,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -484,7 +419,7 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
if (sched_setscheduler(0, SCHED_FIFO, &param)) { if (sched_setscheduler(0, SCHED_FIFO, &param)) {
pr_err("Could not set realtime priority.\n"); pr_err("Could not set realtime priority.\n");
err = -1; err = -1;
goto out_delete_session; goto out_child;
} }
} }
...@@ -512,13 +447,15 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -512,13 +447,15 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
if (record__mmap_read_all(rec) < 0) { if (record__mmap_read_all(rec) < 0) {
err = -1; err = -1;
goto out_delete_session; goto out_child;
} }
if (hits == rec->samples) { if (hits == rec->samples) {
if (done) if (done)
break; break;
err = poll(rec->evlist->pollfd, rec->evlist->nr_fds, -1); err = poll(rec->evlist->pollfd, rec->evlist->nr_fds, -1);
if (err < 0 && errno == EINTR)
err = 0;
waking++; waking++;
} }
...@@ -538,12 +475,10 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -538,12 +475,10 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
const char *emsg = strerror_r(workload_exec_errno, msg, sizeof(msg)); const char *emsg = strerror_r(workload_exec_errno, msg, sizeof(msg));
pr_err("Workload failed: %s\n", emsg); pr_err("Workload failed: %s\n", emsg);
err = -1; err = -1;
goto out_delete_session; goto out_child;
} }
if (quiet || signr == SIGUSR1) if (!quiet) {
return 0;
fprintf(stderr, "[ perf record: Woken up %ld times to write data ]\n", waking); fprintf(stderr, "[ perf record: Woken up %ld times to write data ]\n", waking);
/* /*
...@@ -554,12 +489,38 @@ static int __cmd_record(struct record *rec, int argc, const char **argv) ...@@ -554,12 +489,38 @@ static int __cmd_record(struct record *rec, int argc, const char **argv)
(double)rec->bytes_written / 1024.0 / 1024.0, (double)rec->bytes_written / 1024.0 / 1024.0,
file->path, file->path,
rec->bytes_written / 24); rec->bytes_written / 24);
}
return 0; out_child:
if (forks) {
int exit_status;
if (!child_finished)
kill(rec->evlist->workload.pid, SIGTERM);
wait(&exit_status);
if (err < 0)
status = err;
else if (WIFEXITED(exit_status))
status = WEXITSTATUS(exit_status);
else if (WIFSIGNALED(exit_status))
signr = WTERMSIG(exit_status);
} else
status = err;
if (!err && !file->is_pipe) {
rec->session->header.data_size += rec->bytes_written;
if (!rec->no_buildid)
process_buildids(rec);
perf_session__write_header(rec->session, rec->evlist,
file->fd, true);
}
out_delete_session: out_delete_session:
perf_session__delete(session); perf_session__delete(session);
return err; return status;
} }
#define BRANCH_OPT(n, m) \ #define BRANCH_OPT(n, m) \
...@@ -988,6 +949,7 @@ int cmd_record(int argc, const char **argv, const char *prefix __maybe_unused) ...@@ -988,6 +949,7 @@ int cmd_record(int argc, const char **argv, const char *prefix __maybe_unused)
err = __cmd_record(&record, argc, argv); err = __cmd_record(&record, argc, argv);
out_symbol_exit: out_symbol_exit:
perf_evlist__delete(rec->evlist);
symbol__exit(); symbol__exit();
return err; return err;
} }
...@@ -66,7 +66,7 @@ struct sched_atom { ...@@ -66,7 +66,7 @@ struct sched_atom {
struct task_desc *wakee; struct task_desc *wakee;
}; };
#define TASK_STATE_TO_CHAR_STR "RSDTtZX" #define TASK_STATE_TO_CHAR_STR "RSDTtZXxKWP"
enum thread_state { enum thread_state {
THREAD_SLEEPING = 0, THREAD_SLEEPING = 0,
...@@ -1300,17 +1300,25 @@ static int map_switch_event(struct perf_sched *sched, struct perf_evsel *evsel, ...@@ -1300,17 +1300,25 @@ static int map_switch_event(struct perf_sched *sched, struct perf_evsel *evsel,
new_shortname = 0; new_shortname = 0;
if (!sched_in->shortname[0]) { if (!sched_in->shortname[0]) {
if (!strcmp(thread__comm_str(sched_in), "swapper")) {
/*
* Don't allocate a letter-number for swapper:0
* as a shortname. Instead, we use '.' for it.
*/
sched_in->shortname[0] = '.';
sched_in->shortname[1] = ' ';
} else {
sched_in->shortname[0] = sched->next_shortname1; sched_in->shortname[0] = sched->next_shortname1;
sched_in->shortname[1] = sched->next_shortname2; sched_in->shortname[1] = sched->next_shortname2;
if (sched->next_shortname1 < 'Z') { if (sched->next_shortname1 < 'Z') {
sched->next_shortname1++; sched->next_shortname1++;
} else { } else {
sched->next_shortname1='A'; sched->next_shortname1 = 'A';
if (sched->next_shortname2 < '9') { if (sched->next_shortname2 < '9')
sched->next_shortname2++; sched->next_shortname2++;
} else { else
sched->next_shortname2='0'; sched->next_shortname2 = '0';
} }
} }
new_shortname = 1; new_shortname = 1;
...@@ -1322,12 +1330,9 @@ static int map_switch_event(struct perf_sched *sched, struct perf_evsel *evsel, ...@@ -1322,12 +1330,9 @@ static int map_switch_event(struct perf_sched *sched, struct perf_evsel *evsel,
else else
printf("*"); printf("*");
if (sched->curr_thread[cpu]) { if (sched->curr_thread[cpu])
if (sched->curr_thread[cpu]->tid)
printf("%2s ", sched->curr_thread[cpu]->shortname); printf("%2s ", sched->curr_thread[cpu]->shortname);
else else
printf(". ");
} else
printf(" "); printf(" ");
} }
...@@ -1635,6 +1640,7 @@ static int __cmd_record(int argc, const char **argv) ...@@ -1635,6 +1640,7 @@ static int __cmd_record(int argc, const char **argv)
"-e", "sched:sched_stat_runtime", "-e", "sched:sched_stat_runtime",
"-e", "sched:sched_process_fork", "-e", "sched:sched_process_fork",
"-e", "sched:sched_wakeup", "-e", "sched:sched_wakeup",
"-e", "sched:sched_wakeup_new",
"-e", "sched:sched_migrate_task", "-e", "sched:sched_migrate_task",
}; };
......
...@@ -174,7 +174,6 @@ CORE_FEATURE_TESTS = \ ...@@ -174,7 +174,6 @@ CORE_FEATURE_TESTS = \
libpython-version \ libpython-version \
libslang \ libslang \
libunwind \ libunwind \
on-exit \
stackprotector-all \ stackprotector-all \
timerfd \ timerfd \
libdw-dwarf-unwind libdw-dwarf-unwind
...@@ -200,7 +199,6 @@ VF_FEATURE_TESTS = \ ...@@ -200,7 +199,6 @@ VF_FEATURE_TESTS = \
libelf-getphdrnum \ libelf-getphdrnum \
libelf-mmap \ libelf-mmap \
libpython-version \ libpython-version \
on-exit \
stackprotector-all \ stackprotector-all \
timerfd \ timerfd \
libunwind-debug-frame \ libunwind-debug-frame \
...@@ -571,12 +569,6 @@ ifneq ($(filter -lbfd,$(EXTLIBS)),) ...@@ -571,12 +569,6 @@ ifneq ($(filter -lbfd,$(EXTLIBS)),)
CFLAGS += -DHAVE_LIBBFD_SUPPORT CFLAGS += -DHAVE_LIBBFD_SUPPORT
endif endif
ifndef NO_ON_EXIT
ifeq ($(feature-on-exit), 1)
CFLAGS += -DHAVE_ON_EXIT_SUPPORT
endif
endif
ifndef NO_BACKTRACE ifndef NO_BACKTRACE
ifeq ($(feature-backtrace), 1) ifeq ($(feature-backtrace), 1)
CFLAGS += -DHAVE_BACKTRACE_SUPPORT CFLAGS += -DHAVE_BACKTRACE_SUPPORT
......
...@@ -24,7 +24,6 @@ FILES= \ ...@@ -24,7 +24,6 @@ FILES= \
test-libslang.bin \ test-libslang.bin \
test-libunwind.bin \ test-libunwind.bin \
test-libunwind-debug-frame.bin \ test-libunwind-debug-frame.bin \
test-on-exit.bin \
test-stackprotector-all.bin \ test-stackprotector-all.bin \
test-timerfd.bin \ test-timerfd.bin \
test-libdw-dwarf-unwind.bin test-libdw-dwarf-unwind.bin
...@@ -133,9 +132,6 @@ test-liberty-z.bin: ...@@ -133,9 +132,6 @@ test-liberty-z.bin:
test-cplus-demangle.bin: test-cplus-demangle.bin:
$(BUILD) -liberty $(BUILD) -liberty
test-on-exit.bin:
$(BUILD)
test-backtrace.bin: test-backtrace.bin:
$(BUILD) $(BUILD)
......
...@@ -69,10 +69,6 @@ ...@@ -69,10 +69,6 @@
# include "test-libbfd.c" # include "test-libbfd.c"
#undef main #undef main
#define main main_test_on_exit
# include "test-on-exit.c"
#undef main
#define main main_test_backtrace #define main main_test_backtrace
# include "test-backtrace.c" # include "test-backtrace.c"
#undef main #undef main
...@@ -110,7 +106,6 @@ int main(int argc, char *argv[]) ...@@ -110,7 +106,6 @@ int main(int argc, char *argv[])
main_test_gtk2(argc, argv); main_test_gtk2(argc, argv);
main_test_gtk2_infobar(argc, argv); main_test_gtk2_infobar(argc, argv);
main_test_libbfd(); main_test_libbfd();
main_test_on_exit();
main_test_backtrace(); main_test_backtrace();
main_test_libnuma(); main_test_libnuma();
main_test_timerfd(); main_test_timerfd();
......
#include <stdio.h>
#include <stdlib.h>
static void exit_fn(int status, void *__data)
{
printf("exit status: %d, data: %d\n", status, *(int *)__data);
}
static int data = 123;
int main(void)
{
on_exit(exit_fn, &data);
return 321;
}
...@@ -256,7 +256,7 @@ static int process_sample_event(struct machine *machine, ...@@ -256,7 +256,7 @@ static int process_sample_event(struct machine *machine,
return -1; return -1;
} }
thread = machine__findnew_thread(machine, sample.pid, sample.pid); thread = machine__findnew_thread(machine, sample.pid, sample.tid);
if (!thread) { if (!thread) {
pr_debug("machine__findnew_thread failed\n"); pr_debug("machine__findnew_thread failed\n");
return -1; return -1;
......
...@@ -69,6 +69,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine) ...@@ -69,6 +69,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine)
evsel->hists.symbol_filter_str = NULL; evsel->hists.symbol_filter_str = NULL;
sample.pid = fake_samples[i].pid; sample.pid = fake_samples[i].pid;
sample.tid = fake_samples[i].pid;
sample.ip = fake_samples[i].ip; sample.ip = fake_samples[i].ip;
if (perf_event__preprocess_sample(&event, machine, &al, if (perf_event__preprocess_sample(&event, machine, &al,
......
...@@ -81,6 +81,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine) ...@@ -81,6 +81,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine)
}; };
sample.pid = fake_common_samples[k].pid; sample.pid = fake_common_samples[k].pid;
sample.tid = fake_common_samples[k].pid;
sample.ip = fake_common_samples[k].ip; sample.ip = fake_common_samples[k].ip;
if (perf_event__preprocess_sample(&event, machine, &al, if (perf_event__preprocess_sample(&event, machine, &al,
&sample) < 0) &sample) < 0)
...@@ -104,6 +105,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine) ...@@ -104,6 +105,7 @@ static int add_hist_entries(struct perf_evlist *evlist, struct machine *machine)
}; };
sample.pid = fake_samples[i][k].pid; sample.pid = fake_samples[i][k].pid;
sample.tid = fake_samples[i][k].pid;
sample.ip = fake_samples[i][k].ip; sample.ip = fake_samples[i][k].ip;
if (perf_event__preprocess_sample(&event, machine, &al, if (perf_event__preprocess_sample(&event, machine, &al,
&sample) < 0) &sample) < 0)
......
...@@ -25,7 +25,7 @@ int build_id__mark_dso_hit(struct perf_tool *tool __maybe_unused, ...@@ -25,7 +25,7 @@ int build_id__mark_dso_hit(struct perf_tool *tool __maybe_unused,
struct addr_location al; struct addr_location al;
u8 cpumode = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK; u8 cpumode = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;
struct thread *thread = machine__findnew_thread(machine, sample->pid, struct thread *thread = machine__findnew_thread(machine, sample->pid,
sample->pid); sample->tid);
if (thread == NULL) { if (thread == NULL) {
pr_err("problem processing %d event, skipping it.\n", pr_err("problem processing %d event, skipping it.\n",
......
...@@ -788,7 +788,7 @@ int perf_event__preprocess_sample(const union perf_event *event, ...@@ -788,7 +788,7 @@ int perf_event__preprocess_sample(const union perf_event *event,
{ {
u8 cpumode = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK; u8 cpumode = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;
struct thread *thread = machine__findnew_thread(machine, sample->pid, struct thread *thread = machine__findnew_thread(machine, sample->pid,
sample->pid); sample->tid);
if (thread == NULL) if (thread == NULL)
return -1; return -1;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册