aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arc
diff options
context:
space:
mode:
Diffstat (limited to 'arch/arc')
-rw-r--r--arch/arc/include/asm/perf_event.h53
-rw-r--r--arch/arc/kernel/perf_event.c18
2 files changed, 37 insertions, 34 deletions
diff --git a/arch/arc/include/asm/perf_event.h b/arch/arc/include/asm/perf_event.h
index cbf755e32a03..1c45667c1367 100644
--- a/arch/arc/include/asm/perf_event.h
+++ b/arch/arc/include/asm/perf_event.h
@@ -57,26 +57,7 @@ struct arc_reg_cc_build {
57#define PERF_COUNT_ARC_HW_MAX (PERF_COUNT_HW_MAX + 6) 57#define PERF_COUNT_ARC_HW_MAX (PERF_COUNT_HW_MAX + 6)
58 58
59/* 59/*
60 * The "generalized" performance events seem to really be a copy 60 * Some ARC pct quirks:
61 * of the available events on x86 processors; the mapping to ARC
62 * events is not always possible 1-to-1. Fortunately, there doesn't
63 * seem to be an exact definition for these events, so we can cheat
64 * a bit where necessary.
65 *
66 * In particular, the following PERF events may behave a bit differently
67 * compared to other architectures:
68 *
69 * PERF_COUNT_HW_CPU_CYCLES
70 * Cycles not in halted state
71 *
72 * PERF_COUNT_HW_REF_CPU_CYCLES
73 * Reference cycles not in halted state, same as PERF_COUNT_HW_CPU_CYCLES
74 * for now as we don't do Dynamic Voltage/Frequency Scaling (yet)
75 *
76 * PERF_COUNT_HW_BUS_CYCLES
77 * Unclear what this means, Intel uses 0x013c, which according to
78 * their datasheet means "unhalted reference cycles". It sounds similar
79 * to PERF_COUNT_HW_REF_CPU_CYCLES, and we use the same counter for it.
80 * 61 *
81 * PERF_COUNT_HW_STALLED_CYCLES_BACKEND 62 * PERF_COUNT_HW_STALLED_CYCLES_BACKEND
82 * PERF_COUNT_HW_STALLED_CYCLES_FRONTEND 63 * PERF_COUNT_HW_STALLED_CYCLES_FRONTEND
@@ -91,21 +72,35 @@ struct arc_reg_cc_build {
91 * Note that I$ cache misses aren't counted by either of the two! 72 * Note that I$ cache misses aren't counted by either of the two!
92 */ 73 */
93 74
75/*
76 * ARC PCT has hardware conditions with fixed "names" but variable "indexes"
77 * (based on a specific RTL build)
78 * Below is the static map between perf generic/arc specific event_id and
79 * h/w condition names.
80 * At the time of probe, we loop thru each index and find it's name to
81 * complete the mapping of perf event_id to h/w index as latter is needed
82 * to program the counter really
83 */
94static const char * const arc_pmu_ev_hw_map[] = { 84static const char * const arc_pmu_ev_hw_map[] = {
85 /* count cycles */
95 [PERF_COUNT_HW_CPU_CYCLES] = "crun", 86 [PERF_COUNT_HW_CPU_CYCLES] = "crun",
96 [PERF_COUNT_HW_REF_CPU_CYCLES] = "crun", 87 [PERF_COUNT_HW_REF_CPU_CYCLES] = "crun",
97 [PERF_COUNT_HW_BUS_CYCLES] = "crun", 88 [PERF_COUNT_HW_BUS_CYCLES] = "crun",
98 [PERF_COUNT_HW_INSTRUCTIONS] = "iall", 89
99 [PERF_COUNT_HW_BRANCH_MISSES] = "bpfail",
100 [PERF_COUNT_HW_BRANCH_INSTRUCTIONS] = "ijmp",
101 [PERF_COUNT_HW_STALLED_CYCLES_FRONTEND] = "bflush", 90 [PERF_COUNT_HW_STALLED_CYCLES_FRONTEND] = "bflush",
102 [PERF_COUNT_HW_STALLED_CYCLES_BACKEND] = "bstall", 91 [PERF_COUNT_HW_STALLED_CYCLES_BACKEND] = "bstall",
103 [PERF_COUNT_ARC_DCLM] = "dclm", 92
104 [PERF_COUNT_ARC_DCSM] = "dcsm", 93 /* counts condition */
105 [PERF_COUNT_ARC_ICM] = "icm", 94 [PERF_COUNT_HW_INSTRUCTIONS] = "iall",
106 [PERF_COUNT_ARC_BPOK] = "bpok", 95 [PERF_COUNT_HW_BRANCH_INSTRUCTIONS] = "ijmp",
107 [PERF_COUNT_ARC_EDTLB] = "edtlb", 96 [PERF_COUNT_ARC_BPOK] = "bpok", /* NP-NT, PT-T, PNT-NT */
108 [PERF_COUNT_ARC_EITLB] = "eitlb", 97 [PERF_COUNT_HW_BRANCH_MISSES] = "bpfail", /* NP-T, PT-NT, PNT-T */
98
99 [PERF_COUNT_ARC_DCLM] = "dclm", /* D-cache Load Miss */
100 [PERF_COUNT_ARC_DCSM] = "dcsm", /* D-cache Store Miss */
101 [PERF_COUNT_ARC_ICM] = "icm", /* I-cache Miss */
102 [PERF_COUNT_ARC_EDTLB] = "edtlb", /* D-TLB Miss */
103 [PERF_COUNT_ARC_EITLB] = "eitlb", /* I-TLB Miss */
109}; 104};
110 105
111#define C(_x) PERF_COUNT_HW_CACHE_##_x 106#define C(_x) PERF_COUNT_HW_CACHE_##_x
diff --git a/arch/arc/kernel/perf_event.c b/arch/arc/kernel/perf_event.c
index 64261c2711b1..181baeed4495 100644
--- a/arch/arc/kernel/perf_event.c
+++ b/arch/arc/kernel/perf_event.c
@@ -90,6 +90,10 @@ static int arc_pmu_cache_event(u64 config)
90 if (ret == CACHE_OP_UNSUPPORTED) 90 if (ret == CACHE_OP_UNSUPPORTED)
91 return -ENOENT; 91 return -ENOENT;
92 92
93 pr_debug("init cache event: type/op/result %d/%d/%d with h/w %d \'%s\'\n",
94 cache_type, cache_op, cache_result, ret,
95 arc_pmu_ev_hw_map[ret]);
96
93 return ret; 97 return ret;
94} 98}
95 99
@@ -106,8 +110,9 @@ static int arc_pmu_event_init(struct perf_event *event)
106 if (arc_pmu->ev_hw_idx[event->attr.config] < 0) 110 if (arc_pmu->ev_hw_idx[event->attr.config] < 0)
107 return -ENOENT; 111 return -ENOENT;
108 hwc->config = arc_pmu->ev_hw_idx[event->attr.config]; 112 hwc->config = arc_pmu->ev_hw_idx[event->attr.config];
109 pr_debug("initializing event %d with cfg %d\n", 113 pr_debug("init event %d with h/w %d \'%s\'\n",
110 (int) event->attr.config, (int) hwc->config); 114 (int) event->attr.config, (int) hwc->config,
115 arc_pmu_ev_hw_map[event->attr.config]);
111 return 0; 116 return 0;
112 case PERF_TYPE_HW_CACHE: 117 case PERF_TYPE_HW_CACHE:
113 ret = arc_pmu_cache_event(event->attr.config); 118 ret = arc_pmu_cache_event(event->attr.config);
@@ -260,19 +265,22 @@ static int arc_pmu_device_probe(struct platform_device *pdev)
260 arc_pmu->n_counters, arc_pmu->counter_size, cc_bcr.c); 265 arc_pmu->n_counters, arc_pmu->counter_size, cc_bcr.c);
261 266
262 cc_name.str[8] = 0; 267 cc_name.str[8] = 0;
263 for (i = 0; i < PERF_COUNT_HW_MAX; i++) 268 for (i = 0; i < PERF_COUNT_ARC_HW_MAX; i++)
264 arc_pmu->ev_hw_idx[i] = -1; 269 arc_pmu->ev_hw_idx[i] = -1;
265 270
271 /* loop thru all available h/w condition indexes */
266 for (j = 0; j < cc_bcr.c; j++) { 272 for (j = 0; j < cc_bcr.c; j++) {
267 write_aux_reg(ARC_REG_CC_INDEX, j); 273 write_aux_reg(ARC_REG_CC_INDEX, j);
268 cc_name.indiv.word0 = read_aux_reg(ARC_REG_CC_NAME0); 274 cc_name.indiv.word0 = read_aux_reg(ARC_REG_CC_NAME0);
269 cc_name.indiv.word1 = read_aux_reg(ARC_REG_CC_NAME1); 275 cc_name.indiv.word1 = read_aux_reg(ARC_REG_CC_NAME1);
276
277 /* See if it has been mapped to a perf event_id */
270 for (i = 0; i < ARRAY_SIZE(arc_pmu_ev_hw_map); i++) { 278 for (i = 0; i < ARRAY_SIZE(arc_pmu_ev_hw_map); i++) {
271 if (arc_pmu_ev_hw_map[i] && 279 if (arc_pmu_ev_hw_map[i] &&
272 !strcmp(arc_pmu_ev_hw_map[i], cc_name.str) && 280 !strcmp(arc_pmu_ev_hw_map[i], cc_name.str) &&
273 strlen(arc_pmu_ev_hw_map[i])) { 281 strlen(arc_pmu_ev_hw_map[i])) {
274 pr_debug("mapping %d to idx %d with name %s\n", 282 pr_debug("mapping perf event %2d to h/w event \'%8s\' (idx %d)\n",
275 i, j, cc_name.str); 283 i, cc_name.str, j);
276 arc_pmu->ev_hw_idx[i] = j; 284 arc_pmu->ev_hw_idx[i] = j;
277 } 285 }
278 } 286 }