Commit e78dbc80 authored by Michael Neuling's avatar Michael Neuling Committed by Paul Mackerras

[PATCH] powerpc: oprofile support for POWER6

POWER6 moves some of the MMCRA bits and also requires some bits to be
cleared each PMU interrupt.
Signed-off-by: default avatarMichael Neuling <mikey@neuling.org>
Acked-by: default avatarAnton Blanchard <anton@samba.org>
Signed-off-by: default avatarPaul Mackerras <paulus@samba.org>
parent 8eb6c6e3
...@@ -237,6 +237,11 @@ struct cpu_spec cpu_specs[] = { ...@@ -237,6 +237,11 @@ struct cpu_spec cpu_specs[] = {
.num_pmcs = 6, .num_pmcs = 6,
.oprofile_cpu_type = "ppc64/power5", .oprofile_cpu_type = "ppc64/power5",
.oprofile_type = PPC_OPROFILE_POWER4, .oprofile_type = PPC_OPROFILE_POWER4,
/* SIHV / SIPR bits are implemented on POWER4+ (GQ)
* and above but only works on POWER5 and above
*/
.oprofile_mmcra_sihv = MMCRA_SIHV,
.oprofile_mmcra_sipr = MMCRA_SIPR,
.platform = "power5", .platform = "power5",
}, },
{ /* Power5 GS */ { /* Power5 GS */
...@@ -250,6 +255,8 @@ struct cpu_spec cpu_specs[] = { ...@@ -250,6 +255,8 @@ struct cpu_spec cpu_specs[] = {
.num_pmcs = 6, .num_pmcs = 6,
.oprofile_cpu_type = "ppc64/power5+", .oprofile_cpu_type = "ppc64/power5+",
.oprofile_type = PPC_OPROFILE_POWER4, .oprofile_type = PPC_OPROFILE_POWER4,
.oprofile_mmcra_sihv = MMCRA_SIHV,
.oprofile_mmcra_sipr = MMCRA_SIPR,
.platform = "power5+", .platform = "power5+",
}, },
{ /* Power6 */ { /* Power6 */
...@@ -260,9 +267,13 @@ struct cpu_spec cpu_specs[] = { ...@@ -260,9 +267,13 @@ struct cpu_spec cpu_specs[] = {
.cpu_user_features = COMMON_USER_POWER6, .cpu_user_features = COMMON_USER_POWER6,
.icache_bsize = 128, .icache_bsize = 128,
.dcache_bsize = 128, .dcache_bsize = 128,
.num_pmcs = 6, .num_pmcs = 8,
.oprofile_cpu_type = "ppc64/power6", .oprofile_cpu_type = "ppc64/power6",
.oprofile_type = PPC_OPROFILE_POWER4, .oprofile_type = PPC_OPROFILE_POWER4,
.oprofile_mmcra_sihv = POWER6_MMCRA_SIHV,
.oprofile_mmcra_sipr = POWER6_MMCRA_SIPR,
.oprofile_mmcra_clear = POWER6_MMCRA_THRM |
POWER6_MMCRA_OTHER,
.platform = "power6", .platform = "power6",
}, },
{ /* Cell Broadband Engine */ { /* Cell Broadband Engine */
......
...@@ -24,10 +24,6 @@ ...@@ -24,10 +24,6 @@
static unsigned long reset_value[OP_MAX_COUNTER]; static unsigned long reset_value[OP_MAX_COUNTER];
static int oprofile_running; static int oprofile_running;
static int mmcra_has_sihv;
/* Unfortunately these bits vary between CPUs */
static unsigned long mmcra_sihv = MMCRA_SIHV;
static unsigned long mmcra_sipr = MMCRA_SIPR;
/* mmcr values are set in power4_reg_setup, used in power4_cpu_setup */ /* mmcr values are set in power4_reg_setup, used in power4_cpu_setup */
static u32 mmcr0_val; static u32 mmcr0_val;
...@@ -40,16 +36,6 @@ static void power4_reg_setup(struct op_counter_config *ctr, ...@@ -40,16 +36,6 @@ static void power4_reg_setup(struct op_counter_config *ctr,
{ {
int i; int i;
/*
* SIHV / SIPR bits are only implemented on POWER4+ (GQ) and above.
* However we disable it on all POWER4 until we verify it works
* (I was seeing some strange behaviour last time I tried).
*
* It has been verified to work on POWER5 so we enable it there.
*/
if (cpu_has_feature(CPU_FTR_MMCRA_SIHV))
mmcra_has_sihv = 1;
/* /*
* The performance counter event settings are given in the mmcr0, * The performance counter event settings are given in the mmcr0,
* mmcr1 and mmcra values passed from the user in the * mmcr1 and mmcra values passed from the user in the
...@@ -202,18 +188,19 @@ static unsigned long get_pc(struct pt_regs *regs) ...@@ -202,18 +188,19 @@ static unsigned long get_pc(struct pt_regs *regs)
unsigned long mmcra; unsigned long mmcra;
/* Cant do much about it */ /* Cant do much about it */
if (!mmcra_has_sihv) if (!cur_cpu_spec->oprofile_mmcra_sihv)
return pc; return pc;
mmcra = mfspr(SPRN_MMCRA); mmcra = mfspr(SPRN_MMCRA);
/* Were we in the hypervisor? */ /* Were we in the hypervisor? */
if (firmware_has_feature(FW_FEATURE_LPAR) && (mmcra & mmcra_sihv)) if (firmware_has_feature(FW_FEATURE_LPAR) &&
(mmcra & cur_cpu_spec->oprofile_mmcra_sihv))
/* function descriptor madness */ /* function descriptor madness */
return *((unsigned long *)hypervisor_bucket); return *((unsigned long *)hypervisor_bucket);
/* We were in userspace, nothing to do */ /* We were in userspace, nothing to do */
if (mmcra & mmcra_sipr) if (mmcra & cur_cpu_spec->oprofile_mmcra_sipr)
return pc; return pc;
#ifdef CONFIG_PPC_RTAS #ifdef CONFIG_PPC_RTAS
...@@ -235,15 +222,14 @@ static unsigned long get_pc(struct pt_regs *regs) ...@@ -235,15 +222,14 @@ static unsigned long get_pc(struct pt_regs *regs)
return pc; return pc;
} }
static int get_kernel(unsigned long pc) static int get_kernel(unsigned long pc, unsigned long mmcra)
{ {
int is_kernel; int is_kernel;
if (!mmcra_has_sihv) { if (!cur_cpu_spec->oprofile_mmcra_sihv) {
is_kernel = is_kernel_addr(pc); is_kernel = is_kernel_addr(pc);
} else { } else {
unsigned long mmcra = mfspr(SPRN_MMCRA); is_kernel = ((mmcra & cur_cpu_spec->oprofile_mmcra_sipr) == 0);
is_kernel = ((mmcra & mmcra_sipr) == 0);
} }
return is_kernel; return is_kernel;
...@@ -257,9 +243,12 @@ static void power4_handle_interrupt(struct pt_regs *regs, ...@@ -257,9 +243,12 @@ static void power4_handle_interrupt(struct pt_regs *regs,
int val; int val;
int i; int i;
unsigned int mmcr0; unsigned int mmcr0;
unsigned long mmcra;
mmcra = mfspr(SPRN_MMCRA);
pc = get_pc(regs); pc = get_pc(regs);
is_kernel = get_kernel(pc); is_kernel = get_kernel(pc, mmcra);
/* set the PMM bit (see comment below) */ /* set the PMM bit (see comment below) */
mtmsrd(mfmsr() | MSR_PMM); mtmsrd(mfmsr() | MSR_PMM);
...@@ -287,6 +276,10 @@ static void power4_handle_interrupt(struct pt_regs *regs, ...@@ -287,6 +276,10 @@ static void power4_handle_interrupt(struct pt_regs *regs,
*/ */
mmcr0 &= ~MMCR0_PMAO; mmcr0 &= ~MMCR0_PMAO;
/* Clear the appropriate bits in the MMCRA */
mmcra &= ~cur_cpu_spec->oprofile_mmcra_clear;
mtspr(SPRN_MMCRA, mmcra);
/* /*
* now clear the freeze bit, counting will not start until we * now clear the freeze bit, counting will not start until we
* rfid from this exception, because only at that point will * rfid from this exception, because only at that point will
......
...@@ -69,6 +69,13 @@ struct cpu_spec { ...@@ -69,6 +69,13 @@ struct cpu_spec {
/* Processor specific oprofile operations */ /* Processor specific oprofile operations */
enum powerpc_oprofile_type oprofile_type; enum powerpc_oprofile_type oprofile_type;
/* Bit locations inside the mmcra change */
unsigned long oprofile_mmcra_sihv;
unsigned long oprofile_mmcra_sipr;
/* Bits to clear during an oprofile exception */
unsigned long oprofile_mmcra_clear;
/* Name of processor class, for the ELF AT_PLATFORM entry */ /* Name of processor class, for the ELF AT_PLATFORM entry */
char *platform; char *platform;
}; };
...@@ -117,7 +124,6 @@ extern void do_cpu_ftr_fixups(unsigned long offset); ...@@ -117,7 +124,6 @@ extern void do_cpu_ftr_fixups(unsigned long offset);
#define CPU_FTR_SMT ASM_CONST(0x0000010000000000) #define CPU_FTR_SMT ASM_CONST(0x0000010000000000)
#define CPU_FTR_COHERENT_ICACHE ASM_CONST(0x0000020000000000) #define CPU_FTR_COHERENT_ICACHE ASM_CONST(0x0000020000000000)
#define CPU_FTR_LOCKLESS_TLBIE ASM_CONST(0x0000040000000000) #define CPU_FTR_LOCKLESS_TLBIE ASM_CONST(0x0000040000000000)
#define CPU_FTR_MMCRA_SIHV ASM_CONST(0x0000080000000000)
#define CPU_FTR_CI_LARGE_PAGE ASM_CONST(0x0000100000000000) #define CPU_FTR_CI_LARGE_PAGE ASM_CONST(0x0000100000000000)
#define CPU_FTR_PAUSE_ZERO ASM_CONST(0x0000200000000000) #define CPU_FTR_PAUSE_ZERO ASM_CONST(0x0000200000000000)
#define CPU_FTR_PURR ASM_CONST(0x0000400000000000) #define CPU_FTR_PURR ASM_CONST(0x0000400000000000)
...@@ -134,7 +140,6 @@ extern void do_cpu_ftr_fixups(unsigned long offset); ...@@ -134,7 +140,6 @@ extern void do_cpu_ftr_fixups(unsigned long offset);
#define CPU_FTR_SMT ASM_CONST(0x0) #define CPU_FTR_SMT ASM_CONST(0x0)
#define CPU_FTR_COHERENT_ICACHE ASM_CONST(0x0) #define CPU_FTR_COHERENT_ICACHE ASM_CONST(0x0)
#define CPU_FTR_LOCKLESS_TLBIE ASM_CONST(0x0) #define CPU_FTR_LOCKLESS_TLBIE ASM_CONST(0x0)
#define CPU_FTR_MMCRA_SIHV ASM_CONST(0x0)
#define CPU_FTR_CI_LARGE_PAGE ASM_CONST(0x0) #define CPU_FTR_CI_LARGE_PAGE ASM_CONST(0x0)
#define CPU_FTR_PURR ASM_CONST(0x0) #define CPU_FTR_PURR ASM_CONST(0x0)
#endif #endif
...@@ -320,7 +325,7 @@ extern void do_cpu_ftr_fixups(unsigned long offset); ...@@ -320,7 +325,7 @@ extern void do_cpu_ftr_fixups(unsigned long offset);
CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | \ CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | \
CPU_FTR_MMCRA | CPU_FTR_SMT | \ CPU_FTR_MMCRA | CPU_FTR_SMT | \
CPU_FTR_COHERENT_ICACHE | CPU_FTR_LOCKLESS_TLBIE | \ CPU_FTR_COHERENT_ICACHE | CPU_FTR_LOCKLESS_TLBIE | \
CPU_FTR_MMCRA_SIHV | CPU_FTR_PURR) CPU_FTR_PURR)
#define CPU_FTRS_POWER6 (CPU_FTR_SPLIT_ID_CACHE | CPU_FTR_USE_TB | \ #define CPU_FTRS_POWER6 (CPU_FTR_SPLIT_ID_CACHE | CPU_FTR_USE_TB | \
CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | \ CPU_FTR_HPTE_TABLE | CPU_FTR_PPCAS_ARCH_V2 | \
CPU_FTR_MMCRA | CPU_FTR_SMT | \ CPU_FTR_MMCRA | CPU_FTR_SMT | \
......
...@@ -443,6 +443,10 @@ ...@@ -443,6 +443,10 @@
#define MMCRA_SIHV 0x10000000UL /* state of MSR HV when SIAR set */ #define MMCRA_SIHV 0x10000000UL /* state of MSR HV when SIAR set */
#define MMCRA_SIPR 0x08000000UL /* state of MSR PR when SIAR set */ #define MMCRA_SIPR 0x08000000UL /* state of MSR PR when SIAR set */
#define MMCRA_SAMPLE_ENABLE 0x00000001UL /* enable sampling */ #define MMCRA_SAMPLE_ENABLE 0x00000001UL /* enable sampling */
#define POWER6_MMCRA_SIHV 0x0000040000000000ULL
#define POWER6_MMCRA_SIPR 0x0000020000000000ULL
#define POWER6_MMCRA_THRM 0x00000020UL
#define POWER6_MMCRA_OTHER 0x0000000EUL
#define SPRN_PMC1 787 #define SPRN_PMC1 787
#define SPRN_PMC2 788 #define SPRN_PMC2 788
#define SPRN_PMC3 789 #define SPRN_PMC3 789
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment