forked from torvalds/linux
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
s390/perf: add callback to perf to enable using AUX buffer
Perf tool need implement a callback to enable using AUX buffer. Perf will do another mmap() to trigger the setup of AUX buffer in kernel if there is such callback. The default size of the AUX buffer is set properly according to the sampling frequency to avoid overflow. It could also be manually set by -m option of perf. The interface of perf is not changed. Diagnostic mode sampling could be started by `perf record -e rBD000` like before. Signed-off-by: Pu Hou <[email protected]> Reviewed-by: Hendrik Brueckner <[email protected]> Signed-off-by: Martin Schwidefsky <[email protected]>
- Loading branch information
Pu Hou
authored and
Martin Schwidefsky
committed
Nov 16, 2017
1 parent
cbf6948
commit a3f22d5
Showing
2 changed files
with
120 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,118 @@ | ||
#include <stdbool.h> | ||
#include <linux/kernel.h> | ||
#include <linux/types.h> | ||
#include <linux/bitops.h> | ||
#include <linux/log2.h> | ||
|
||
#include "../../util/evlist.h" | ||
#include "../../util/auxtrace.h" | ||
#include "../../util/evsel.h" | ||
|
||
#define PERF_EVENT_CPUM_SF 0xB0000 /* Event: Basic-sampling */ | ||
#define PERF_EVENT_CPUM_SF_DIAG 0xBD000 /* Event: Combined-sampling */ | ||
#define DEFAULT_AUX_PAGES 128 | ||
#define DEFAULT_FREQ 4000 | ||
|
||
static void cpumsf_free(struct auxtrace_record *itr) | ||
{ | ||
free(itr); | ||
} | ||
|
||
static size_t cpumsf_info_priv_size(struct auxtrace_record *itr __maybe_unused, | ||
struct perf_evlist *evlist __maybe_unused) | ||
{ | ||
return 0; | ||
} | ||
|
||
static int | ||
cpumsf_info_fill(struct auxtrace_record *itr __maybe_unused, | ||
struct perf_session *session __maybe_unused, | ||
struct auxtrace_info_event *auxtrace_info __maybe_unused, | ||
size_t priv_size __maybe_unused) | ||
{ | ||
return 0; | ||
} | ||
|
||
static unsigned long | ||
cpumsf_reference(struct auxtrace_record *itr __maybe_unused) | ||
{ | ||
return 0; | ||
} | ||
|
||
static int | ||
cpumsf_recording_options(struct auxtrace_record *ar __maybe_unused, | ||
struct perf_evlist *evlist __maybe_unused, | ||
struct record_opts *opts) | ||
{ | ||
unsigned int factor = 1; | ||
unsigned int pages; | ||
|
||
opts->full_auxtrace = true; | ||
|
||
/* | ||
* The AUX buffer size should be set properly to avoid | ||
* overflow of samples if it is not set explicitly. | ||
* DEFAULT_AUX_PAGES is an proper size when sampling frequency | ||
* is DEFAULT_FREQ. It is expected to hold about 1/2 second | ||
* of sampling data. The size used for AUX buffer will scale | ||
* according to the specified frequency and DEFAULT_FREQ. | ||
*/ | ||
if (!opts->auxtrace_mmap_pages) { | ||
if (opts->user_freq != UINT_MAX) | ||
factor = (opts->user_freq + DEFAULT_FREQ | ||
- 1) / DEFAULT_FREQ; | ||
pages = DEFAULT_AUX_PAGES * factor; | ||
opts->auxtrace_mmap_pages = roundup_pow_of_two(pages); | ||
} | ||
|
||
return 0; | ||
} | ||
|
||
static int | ||
cpumsf_parse_snapshot_options(struct auxtrace_record *itr __maybe_unused, | ||
struct record_opts *opts __maybe_unused, | ||
const char *str __maybe_unused) | ||
{ | ||
return 0; | ||
} | ||
|
||
/* | ||
* auxtrace_record__init is called when perf record | ||
* check if the event really need auxtrace | ||
*/ | ||
struct auxtrace_record *auxtrace_record__init(struct perf_evlist *evlist, | ||
int *err) | ||
{ | ||
struct auxtrace_record *aux; | ||
struct perf_evsel *pos; | ||
int diagnose = 0; | ||
|
||
if (evlist->nr_entries == 0) | ||
return NULL; | ||
|
||
evlist__for_each_entry(evlist, pos) { | ||
if (pos->attr.config == PERF_EVENT_CPUM_SF_DIAG) { | ||
diagnose = 1; | ||
break; | ||
} | ||
} | ||
|
||
if (!diagnose) | ||
return NULL; | ||
|
||
/* sampling in diagnose mode. alloc aux buffer */ | ||
aux = zalloc(sizeof(*aux)); | ||
if (aux == NULL) { | ||
*err = -ENOMEM; | ||
return NULL; | ||
} | ||
|
||
aux->parse_snapshot_options = cpumsf_parse_snapshot_options; | ||
aux->recording_options = cpumsf_recording_options; | ||
aux->info_priv_size = cpumsf_info_priv_size; | ||
aux->info_fill = cpumsf_info_fill; | ||
aux->free = cpumsf_free; | ||
aux->reference = cpumsf_reference; | ||
|
||
return aux; | ||
} |