1 // SPDX-License-Identifier: GPL-2.0
3 * Arm Statistical Profiling Extensions (SPE) support
4 * Copyright (c) 2017-2018, Arm Ltd.
7 #include <linux/kernel.h>
8 #include <linux/types.h>
9 #include <linux/bitops.h>
10 #include <linux/log2.h>
13 #include "../../util/cpumap.h"
14 #include "../../util/evsel.h"
15 #include "../../util/evlist.h"
16 #include "../../util/session.h"
17 #include "../../util/util.h"
18 #include "../../util/pmu.h"
19 #include "../../util/debug.h"
20 #include "../../util/auxtrace.h"
21 #include "../../util/arm-spe.h"
23 #define KiB(x) ((x) * 1024)
24 #define MiB(x) ((x) * 1024 * 1024)
26 struct arm_spe_recording
{
27 struct auxtrace_record itr
;
28 struct perf_pmu
*arm_spe_pmu
;
29 struct perf_evlist
*evlist
;
33 arm_spe_info_priv_size(struct auxtrace_record
*itr __maybe_unused
,
34 struct perf_evlist
*evlist __maybe_unused
)
36 return ARM_SPE_AUXTRACE_PRIV_SIZE
;
39 static int arm_spe_info_fill(struct auxtrace_record
*itr
,
40 struct perf_session
*session
,
41 struct auxtrace_info_event
*auxtrace_info
,
44 struct arm_spe_recording
*sper
=
45 container_of(itr
, struct arm_spe_recording
, itr
);
46 struct perf_pmu
*arm_spe_pmu
= sper
->arm_spe_pmu
;
48 if (priv_size
!= ARM_SPE_AUXTRACE_PRIV_SIZE
)
51 if (!session
->evlist
->nr_mmaps
)
54 auxtrace_info
->type
= PERF_AUXTRACE_ARM_SPE
;
55 auxtrace_info
->priv
[ARM_SPE_PMU_TYPE
] = arm_spe_pmu
->type
;
60 static int arm_spe_recording_options(struct auxtrace_record
*itr
,
61 struct perf_evlist
*evlist
,
62 struct record_opts
*opts
)
64 struct arm_spe_recording
*sper
=
65 container_of(itr
, struct arm_spe_recording
, itr
);
66 struct perf_pmu
*arm_spe_pmu
= sper
->arm_spe_pmu
;
67 struct perf_evsel
*evsel
, *arm_spe_evsel
= NULL
;
68 bool privileged
= geteuid() == 0 || perf_event_paranoid() < 0;
69 struct perf_evsel
*tracking_evsel
;
72 sper
->evlist
= evlist
;
74 evlist__for_each_entry(evlist
, evsel
) {
75 if (evsel
->attr
.type
== arm_spe_pmu
->type
) {
77 pr_err("There may be only one " ARM_SPE_PMU_NAME
"x event\n");
81 evsel
->attr
.sample_period
= 1;
82 arm_spe_evsel
= evsel
;
83 opts
->full_auxtrace
= true;
87 if (!opts
->full_auxtrace
)
90 /* We are in full trace mode but '-m,xyz' wasn't specified */
91 if (opts
->full_auxtrace
&& !opts
->auxtrace_mmap_pages
) {
93 opts
->auxtrace_mmap_pages
= MiB(4) / page_size
;
95 opts
->auxtrace_mmap_pages
= KiB(128) / page_size
;
96 if (opts
->mmap_pages
== UINT_MAX
)
97 opts
->mmap_pages
= KiB(256) / page_size
;
101 /* Validate auxtrace_mmap_pages */
102 if (opts
->auxtrace_mmap_pages
) {
103 size_t sz
= opts
->auxtrace_mmap_pages
* (size_t)page_size
;
104 size_t min_sz
= KiB(8);
106 if (sz
< min_sz
|| !is_power_of_2(sz
)) {
107 pr_err("Invalid mmap size for ARM SPE: must be at least %zuKiB and a power of 2\n",
115 * To obtain the auxtrace buffer file descriptor, the auxtrace event
118 perf_evlist__to_front(evlist
, arm_spe_evsel
);
120 perf_evsel__set_sample_bit(arm_spe_evsel
, CPU
);
121 perf_evsel__set_sample_bit(arm_spe_evsel
, TIME
);
122 perf_evsel__set_sample_bit(arm_spe_evsel
, TID
);
124 /* Add dummy event to keep tracking */
125 err
= parse_events(evlist
, "dummy:u", NULL
);
129 tracking_evsel
= perf_evlist__last(evlist
);
130 perf_evlist__set_tracking_event(evlist
, tracking_evsel
);
132 tracking_evsel
->attr
.freq
= 0;
133 tracking_evsel
->attr
.sample_period
= 1;
134 perf_evsel__set_sample_bit(tracking_evsel
, TIME
);
135 perf_evsel__set_sample_bit(tracking_evsel
, CPU
);
136 perf_evsel__reset_sample_bit(tracking_evsel
, BRANCH_STACK
);
141 static u64
arm_spe_reference(struct auxtrace_record
*itr __maybe_unused
)
145 clock_gettime(CLOCK_MONOTONIC_RAW
, &ts
);
147 return ts
.tv_sec
^ ts
.tv_nsec
;
150 static void arm_spe_recording_free(struct auxtrace_record
*itr
)
152 struct arm_spe_recording
*sper
=
153 container_of(itr
, struct arm_spe_recording
, itr
);
158 static int arm_spe_read_finish(struct auxtrace_record
*itr
, int idx
)
160 struct arm_spe_recording
*sper
=
161 container_of(itr
, struct arm_spe_recording
, itr
);
162 struct perf_evsel
*evsel
;
164 evlist__for_each_entry(sper
->evlist
, evsel
) {
165 if (evsel
->attr
.type
== sper
->arm_spe_pmu
->type
)
166 return perf_evlist__enable_event_idx(sper
->evlist
,
172 struct auxtrace_record
*arm_spe_recording_init(int *err
,
173 struct perf_pmu
*arm_spe_pmu
)
175 struct arm_spe_recording
*sper
;
182 sper
= zalloc(sizeof(struct arm_spe_recording
));
188 sper
->arm_spe_pmu
= arm_spe_pmu
;
189 sper
->itr
.recording_options
= arm_spe_recording_options
;
190 sper
->itr
.info_priv_size
= arm_spe_info_priv_size
;
191 sper
->itr
.info_fill
= arm_spe_info_fill
;
192 sper
->itr
.free
= arm_spe_recording_free
;
193 sper
->itr
.reference
= arm_spe_reference
;
194 sper
->itr
.read_finish
= arm_spe_read_finish
;
195 sper
->itr
.alignment
= 0;
200 struct perf_event_attr
201 *arm_spe_pmu_default_config(struct perf_pmu
*arm_spe_pmu
)
203 struct perf_event_attr
*attr
;
205 attr
= zalloc(sizeof(struct perf_event_attr
));
207 pr_err("arm_spe default config cannot allocate a perf_event_attr\n");
212 * If kernel driver doesn't advertise a minimum,
213 * use max allowable by PMSIDR_EL1.INTERVAL
215 if (perf_pmu__scan_file(arm_spe_pmu
, "caps/min_interval", "%llu",
216 &attr
->sample_period
) != 1) {
217 pr_debug("arm_spe driver doesn't advertise a min. interval. Using 4096\n");
218 attr
->sample_period
= 4096;
221 arm_spe_pmu
->selectable
= true;
222 arm_spe_pmu
->is_uncore
= false;