2020-05-05 17:49:08 +03:00
/* SPDX-License-Identifier: GPL-2.0 */
# include "perf-sys.h"
# include "util/cloexec.h"
# include "util/evlist.h"
# include "util/evsel.h"
# include "util/parse-events.h"
# include "util/perf_api_probe.h"
# include <perf/cpumap.h>
# include <errno.h>
typedef void ( * setup_probe_fn_t ) ( struct evsel * evsel ) ;
2022-01-05 09:13:51 +03:00
static int perf_do_probe_api ( setup_probe_fn_t fn , struct perf_cpu cpu , const char * str )
2020-05-05 17:49:08 +03:00
{
struct evlist * evlist ;
struct evsel * evsel ;
unsigned long flags = perf_event_open_cloexec_flag ( ) ;
int err = - EAGAIN , fd ;
static pid_t pid = - 1 ;
evlist = evlist__new ( ) ;
if ( ! evlist )
return - ENOMEM ;
2022-08-09 11:07:02 +03:00
if ( parse_event ( evlist , str ) )
2020-05-05 17:49:08 +03:00
goto out_delete ;
evsel = evlist__first ( evlist ) ;
while ( 1 ) {
2022-01-05 09:13:51 +03:00
fd = sys_perf_event_open ( & evsel - > core . attr , pid , cpu . cpu , - 1 , flags ) ;
2020-05-05 17:49:08 +03:00
if ( fd < 0 ) {
if ( pid = = - 1 & & errno = = EACCES ) {
pid = 0 ;
continue ;
}
goto out_delete ;
}
break ;
}
close ( fd ) ;
fn ( evsel ) ;
2022-01-05 09:13:51 +03:00
fd = sys_perf_event_open ( & evsel - > core . attr , pid , cpu . cpu , - 1 , flags ) ;
2020-05-05 17:49:08 +03:00
if ( fd < 0 ) {
if ( errno = = EINVAL )
err = - EINVAL ;
goto out_delete ;
}
close ( fd ) ;
err = 0 ;
out_delete :
evlist__delete ( evlist ) ;
return err ;
}
static bool perf_probe_api ( setup_probe_fn_t fn )
{
const char * try [ ] = { " cycles:u " , " instructions:u " , " cpu-clock:u " , NULL } ;
struct perf_cpu_map * cpus ;
2022-01-05 09:13:51 +03:00
struct perf_cpu cpu ;
int ret , i = 0 ;
2020-05-05 17:49:08 +03:00
cpus = perf_cpu_map__new ( NULL ) ;
if ( ! cpus )
return false ;
2022-01-22 07:58:10 +03:00
cpu = perf_cpu_map__cpu ( cpus , 0 ) ;
2020-05-05 17:49:08 +03:00
perf_cpu_map__put ( cpus ) ;
do {
ret = perf_do_probe_api ( fn , cpu , try [ i + + ] ) ;
if ( ! ret )
return true ;
} while ( ret = = - EAGAIN & & try [ i ] ) ;
return false ;
}
static void perf_probe_sample_identifier ( struct evsel * evsel )
{
evsel - > core . attr . sample_type | = PERF_SAMPLE_IDENTIFIER ;
}
static void perf_probe_comm_exec ( struct evsel * evsel )
{
evsel - > core . attr . comm_exec = 1 ;
}
static void perf_probe_context_switch ( struct evsel * evsel )
{
evsel - > core . attr . context_switch = 1 ;
}
2020-05-12 15:19:18 +03:00
static void perf_probe_text_poke ( struct evsel * evsel )
{
evsel - > core . attr . text_poke = 1 ;
}
perf record: Add --buildid-mmap option to enable PERF_RECORD_MMAP2's build id
Add --buildid-mmap option to enable build id in PERF_RECORD_MMAP2 events.
It will only work if there's kernel support for that and it disables
build id cache (implies --no-buildid).
It's also possible to enable it permanently via config option in
~/.perfconfig file:
[record]
build-id=mmap
Also added build_id bit in the verbose output for perf_event_attr:
# perf record --buildid-mmap -vv
...
perf_event_attr:
type 1
size 120
...
build_id 1
Adding also missing text_poke bit.
Committer testing:
$ perf record -h build
Usage: perf record [<options>] [<command>]
or: perf record [<options>] -- <command> [<options>]
-B, --no-buildid do not collect buildids in perf.data
-N, --no-buildid-cache
do not update the buildid cache
--buildid-all Record build-id of all DSOs regardless of hits
--buildid-mmap Record build-id in map events
$
$ perf record --buildid-mmap sleep 1
Failed: no support to record build id in mmap events, update your kernel.
$
After adding the needed kernel bits in a test kernel:
$ perf record -vv --buildid-mmap sleep 1 |& grep -m1 build
Enabling build id in mmap2 events.
$ perf evlist -v
cycles:u: size: 120, { sample_period, sample_freq }: 4000, sample_type: IP|TID|TIME|PERIOD, read_format: ID, disabled: 1, inherit: 1, exclude_kernel: 1, mmap: 1, comm: 1, freq: 1, enable_on_exec: 1, task: 1, sample_id_all: 1, exclude_guest: 1, mmap2: 1, comm_exec: 1, ksymbol: 1, bpf_event: 1, build_id: 1
$
Signed-off-by: Jiri Olsa <jolsa@kernel.org>
Tested-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Acked-by: Ian Rogers <irogers@google.com>
Acked-by: Namhyung Kim <namhyung@kernel.org>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com>
Cc: Alexei Budankov <abudankov@huawei.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Mark Rutland <mark.rutland@arm.com>
Cc: Michael Petlan <mpetlan@redhat.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Song Liu <songliubraving@fb.com>
Cc: Stephane Eranian <eranian@google.com>
Link: http://lore.kernel.org/lkml/20201214105457.543111-16-jolsa@kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2020-12-14 13:54:57 +03:00
static void perf_probe_build_id ( struct evsel * evsel )
{
evsel - > core . attr . build_id = 1 ;
}
2021-05-27 21:28:35 +03:00
static void perf_probe_cgroup ( struct evsel * evsel )
{
evsel - > core . attr . cgroup = 1 ;
}
2020-05-05 17:49:08 +03:00
bool perf_can_sample_identifier ( void )
{
return perf_probe_api ( perf_probe_sample_identifier ) ;
}
bool perf_can_comm_exec ( void )
{
return perf_probe_api ( perf_probe_comm_exec ) ;
}
bool perf_can_record_switch_events ( void )
{
return perf_probe_api ( perf_probe_context_switch ) ;
}
2020-05-12 15:19:18 +03:00
bool perf_can_record_text_poke_events ( void )
{
return perf_probe_api ( perf_probe_text_poke ) ;
}
2020-05-05 17:49:08 +03:00
bool perf_can_record_cpu_wide ( void )
{
struct perf_event_attr attr = {
. type = PERF_TYPE_SOFTWARE ,
. config = PERF_COUNT_SW_CPU_CLOCK ,
. exclude_kernel = 1 ,
} ;
struct perf_cpu_map * cpus ;
2022-01-05 09:13:51 +03:00
struct perf_cpu cpu ;
int fd ;
2020-05-05 17:49:08 +03:00
cpus = perf_cpu_map__new ( NULL ) ;
if ( ! cpus )
return false ;
2022-01-05 09:13:51 +03:00
2022-01-22 07:58:10 +03:00
cpu = perf_cpu_map__cpu ( cpus , 0 ) ;
2020-05-05 17:49:08 +03:00
perf_cpu_map__put ( cpus ) ;
2022-01-05 09:13:51 +03:00
fd = sys_perf_event_open ( & attr , - 1 , cpu . cpu , - 1 , 0 ) ;
2020-05-05 17:49:08 +03:00
if ( fd < 0 )
return false ;
close ( fd ) ;
return true ;
}
/*
* Architectures are expected to know if AUX area sampling is supported by the
* hardware . Here we check for kernel support .
*/
bool perf_can_aux_sample ( void )
{
struct perf_event_attr attr = {
. size = sizeof ( struct perf_event_attr ) ,
. exclude_kernel = 1 ,
/*
* Non - zero value causes the kernel to calculate the effective
* attribute size up to that byte .
*/
. aux_sample_size = 1 ,
} ;
int fd ;
fd = sys_perf_event_open ( & attr , - 1 , 0 , - 1 , 0 ) ;
/*
* If the kernel attribute is big enough to contain aux_sample_size
* then we assume that it is supported . We are relying on the kernel to
* validate the attribute size before anything else that could be wrong .
*/
if ( fd < 0 & & errno = = E2BIG )
return false ;
if ( fd > = 0 )
close ( fd ) ;
return true ;
}
perf record: Add --buildid-mmap option to enable PERF_RECORD_MMAP2's build id
Add --buildid-mmap option to enable build id in PERF_RECORD_MMAP2 events.
It will only work if there's kernel support for that and it disables
build id cache (implies --no-buildid).
It's also possible to enable it permanently via config option in
~/.perfconfig file:
[record]
build-id=mmap
Also added build_id bit in the verbose output for perf_event_attr:
# perf record --buildid-mmap -vv
...
perf_event_attr:
type 1
size 120
...
build_id 1
Adding also missing text_poke bit.
Committer testing:
$ perf record -h build
Usage: perf record [<options>] [<command>]
or: perf record [<options>] -- <command> [<options>]
-B, --no-buildid do not collect buildids in perf.data
-N, --no-buildid-cache
do not update the buildid cache
--buildid-all Record build-id of all DSOs regardless of hits
--buildid-mmap Record build-id in map events
$
$ perf record --buildid-mmap sleep 1
Failed: no support to record build id in mmap events, update your kernel.
$
After adding the needed kernel bits in a test kernel:
$ perf record -vv --buildid-mmap sleep 1 |& grep -m1 build
Enabling build id in mmap2 events.
$ perf evlist -v
cycles:u: size: 120, { sample_period, sample_freq }: 4000, sample_type: IP|TID|TIME|PERIOD, read_format: ID, disabled: 1, inherit: 1, exclude_kernel: 1, mmap: 1, comm: 1, freq: 1, enable_on_exec: 1, task: 1, sample_id_all: 1, exclude_guest: 1, mmap2: 1, comm_exec: 1, ksymbol: 1, bpf_event: 1, build_id: 1
$
Signed-off-by: Jiri Olsa <jolsa@kernel.org>
Tested-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Acked-by: Ian Rogers <irogers@google.com>
Acked-by: Namhyung Kim <namhyung@kernel.org>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Alexander Shishkin <alexander.shishkin@linux.intel.com>
Cc: Alexei Budankov <abudankov@huawei.com>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Mark Rutland <mark.rutland@arm.com>
Cc: Michael Petlan <mpetlan@redhat.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Song Liu <songliubraving@fb.com>
Cc: Stephane Eranian <eranian@google.com>
Link: http://lore.kernel.org/lkml/20201214105457.543111-16-jolsa@kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2020-12-14 13:54:57 +03:00
bool perf_can_record_build_id ( void )
{
return perf_probe_api ( perf_probe_build_id ) ;
}
2021-05-27 21:28:35 +03:00
bool perf_can_record_cgroup ( void )
{
return perf_probe_api ( perf_probe_cgroup ) ;
}