2 * Performance events support for SH7750-style performance counters
4 * Copyright (C) 2009 Paul Mundt
6 * This file is subject to the terms and conditions of the GNU General Public
7 * License. See the file "COPYING" in the main directory of this archive
10 #include <linux/kernel.h>
11 #include <linux/init.h>
13 #include <linux/irq.h>
14 #include <linux/perf_event.h>
15 #include <asm/processor.h>
17 #define PM_CR_BASE 0xff000084 /* 16-bit */
18 #define PM_CTR_BASE 0xff100004 /* 32-bit */
20 #define PMCR(n) (PM_CR_BASE + ((n) * 0x04))
21 #define PMCTRH(n) (PM_CTR_BASE + 0x00 + ((n) * 0x08))
22 #define PMCTRL(n) (PM_CTR_BASE + 0x04 + ((n) * 0x08))
24 #define PMCR_PMM_MASK 0x0000003f
26 #define PMCR_CLKF 0x00000100
27 #define PMCR_PMCLR 0x00002000
28 #define PMCR_PMST 0x00004000
29 #define PMCR_PMEN 0x00008000
31 static struct sh_pmu sh7750_pmu
;
34 * There are a number of events supported by each counter (33 in total).
35 * Since we have 2 counters, each counter will take the event code as it
36 * corresponds to the PMCR PMM setting. Each counter can be configured
39 * Event Code Description
40 * ---------- -----------
42 * 0x01 Operand read access
43 * 0x02 Operand write access
45 * 0x04 Operand cache read miss
46 * 0x05 Operand cache write miss
47 * 0x06 Instruction fetch (w/ cache)
48 * 0x07 Instruction TLB miss
49 * 0x08 Instruction cache miss
50 * 0x09 All operand accesses
51 * 0x0a All instruction accesses
52 * 0x0b OC RAM operand access
53 * 0x0d On-chip I/O space access
54 * 0x0e Operand access (r/w)
55 * 0x0f Operand cache miss (r/w)
56 * 0x10 Branch instruction
59 * 0x13 Instruction execution
60 * 0x14 Instruction execution in parallel
61 * 0x15 FPU Instruction execution
64 * 0x18 trapa instruction execution
67 * 0x21 Instruction cache fill
68 * 0x22 Operand cache fill
70 * 0x24 Pipeline freeze by I-cache miss
71 * 0x25 Pipeline freeze by D-cache miss
72 * 0x27 Pipeline freeze by branch instruction
73 * 0x28 Pipeline freeze by CPU register
74 * 0x29 Pipeline freeze by FPU
77 static const int sh7750_general_events
[] = {
78 [PERF_COUNT_HW_CPU_CYCLES
] = 0x0023,
79 [PERF_COUNT_HW_INSTRUCTIONS
] = 0x000a,
80 [PERF_COUNT_HW_CACHE_REFERENCES
] = 0x0006, /* I-cache */
81 [PERF_COUNT_HW_CACHE_MISSES
] = 0x0008, /* I-cache */
82 [PERF_COUNT_HW_BRANCH_INSTRUCTIONS
] = 0x0010,
83 [PERF_COUNT_HW_BRANCH_MISSES
] = -1,
84 [PERF_COUNT_HW_BUS_CYCLES
] = -1,
87 #define C(x) PERF_COUNT_HW_CACHE_##x
89 static const int sh7750_cache_events
90 [PERF_COUNT_HW_CACHE_MAX
]
91 [PERF_COUNT_HW_CACHE_OP_MAX
]
92 [PERF_COUNT_HW_CACHE_RESULT_MAX
] =
96 [ C(RESULT_ACCESS
) ] = 0x0001,
97 [ C(RESULT_MISS
) ] = 0x0004,
100 [ C(RESULT_ACCESS
) ] = 0x0002,
101 [ C(RESULT_MISS
) ] = 0x0005,
103 [ C(OP_PREFETCH
) ] = {
104 [ C(RESULT_ACCESS
) ] = 0,
105 [ C(RESULT_MISS
) ] = 0,
111 [ C(RESULT_ACCESS
) ] = 0x0006,
112 [ C(RESULT_MISS
) ] = 0x0008,
115 [ C(RESULT_ACCESS
) ] = -1,
116 [ C(RESULT_MISS
) ] = -1,
118 [ C(OP_PREFETCH
) ] = {
119 [ C(RESULT_ACCESS
) ] = 0,
120 [ C(RESULT_MISS
) ] = 0,
126 [ C(RESULT_ACCESS
) ] = 0,
127 [ C(RESULT_MISS
) ] = 0,
130 [ C(RESULT_ACCESS
) ] = 0,
131 [ C(RESULT_MISS
) ] = 0,
133 [ C(OP_PREFETCH
) ] = {
134 [ C(RESULT_ACCESS
) ] = 0,
135 [ C(RESULT_MISS
) ] = 0,
141 [ C(RESULT_ACCESS
) ] = 0,
142 [ C(RESULT_MISS
) ] = 0x0003,
145 [ C(RESULT_ACCESS
) ] = 0,
146 [ C(RESULT_MISS
) ] = 0,
148 [ C(OP_PREFETCH
) ] = {
149 [ C(RESULT_ACCESS
) ] = 0,
150 [ C(RESULT_MISS
) ] = 0,
156 [ C(RESULT_ACCESS
) ] = 0,
157 [ C(RESULT_MISS
) ] = 0x0007,
160 [ C(RESULT_ACCESS
) ] = -1,
161 [ C(RESULT_MISS
) ] = -1,
163 [ C(OP_PREFETCH
) ] = {
164 [ C(RESULT_ACCESS
) ] = -1,
165 [ C(RESULT_MISS
) ] = -1,
171 [ C(RESULT_ACCESS
) ] = -1,
172 [ C(RESULT_MISS
) ] = -1,
175 [ C(RESULT_ACCESS
) ] = -1,
176 [ C(RESULT_MISS
) ] = -1,
178 [ C(OP_PREFETCH
) ] = {
179 [ C(RESULT_ACCESS
) ] = -1,
180 [ C(RESULT_MISS
) ] = -1,
185 static int sh7750_event_map(int event
)
187 return sh7750_general_events
[event
];
190 static u64
sh7750_pmu_read(int idx
)
192 return (u64
)((u64
)(__raw_readl(PMCTRH(idx
)) & 0xffff) << 32) |
193 __raw_readl(PMCTRL(idx
));
196 static void sh7750_pmu_disable(struct hw_perf_event
*hwc
, int idx
)
200 tmp
= __raw_readw(PMCR(idx
));
201 tmp
&= ~(PMCR_PMM_MASK
| PMCR_PMEN
);
202 __raw_writew(tmp
, PMCR(idx
));
205 static void sh7750_pmu_enable(struct hw_perf_event
*hwc
, int idx
)
207 __raw_writew(__raw_readw(PMCR(idx
)) | PMCR_PMCLR
, PMCR(idx
));
208 __raw_writew(hwc
->config
| PMCR_PMEN
| PMCR_PMST
, PMCR(idx
));
211 static void sh7750_pmu_disable_all(void)
215 for (i
= 0; i
< sh7750_pmu
.num_events
; i
++)
216 __raw_writew(__raw_readw(PMCR(i
)) & ~PMCR_PMEN
, PMCR(i
));
219 static void sh7750_pmu_enable_all(void)
223 for (i
= 0; i
< sh7750_pmu
.num_events
; i
++)
224 __raw_writew(__raw_readw(PMCR(i
)) | PMCR_PMEN
, PMCR(i
));
227 static struct sh_pmu sh7750_pmu
= {
230 .event_map
= sh7750_event_map
,
231 .max_events
= ARRAY_SIZE(sh7750_general_events
),
232 .raw_event_mask
= PMCR_PMM_MASK
,
233 .cache_events
= &sh7750_cache_events
,
234 .read
= sh7750_pmu_read
,
235 .disable
= sh7750_pmu_disable
,
236 .enable
= sh7750_pmu_enable
,
237 .disable_all
= sh7750_pmu_disable_all
,
238 .enable_all
= sh7750_pmu_enable_all
,
241 static int __init
sh7750_pmu_init(void)
244 * Make sure this CPU actually has perf counters.
246 if (!(boot_cpu_data
.flags
& CPU_HAS_PERF_COUNTER
)) {
247 pr_notice("HW perf events unsupported, software events only.\n");
251 return register_sh_pmu(&sh7750_pmu
);
253 arch_initcall(sh7750_pmu_init
);