| /** |
| * op_model_v7.c |
| * ARM V7 (Cortex A8) Event Monitor Driver |
| * |
| * Copyright 2008 Jean Pihet <jpihet@mvista.com> |
| * Copyright 2004 ARM SMP Development Team |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 as |
| * published by the Free Software Foundation. |
| */ |
| #include <linux/types.h> |
| #include <linux/errno.h> |
| #include <linux/oprofile.h> |
| #include <linux/interrupt.h> |
| #include <linux/irq.h> |
| #include <linux/smp.h> |
| |
| #include "op_counter.h" |
| #include "op_arm_model.h" |
| #include "op_model_v7.h" |
| |
| /* #define DEBUG */ |
| |
| |
| /* |
| * ARM V7 PMNC support |
| */ |
| |
| static u32 cnt_en[CNTMAX]; |
| |
| static inline void armv7_pmnc_write(u32 val) |
| { |
| val &= PMNC_MASK; |
| asm volatile("mcr p15, 0, %0, c9, c12, 0" : : "r" (val)); |
| } |
| |
| static inline u32 armv7_pmnc_read(void) |
| { |
| u32 val; |
| |
| asm volatile("mrc p15, 0, %0, c9, c12, 0" : "=r" (val)); |
| return val; |
| } |
| |
| static inline u32 armv7_pmnc_enable_counter(unsigned int cnt) |
| { |
| u32 val; |
| |
| if (cnt >= CNTMAX) { |
| printk(KERN_ERR "oprofile: CPU%u enabling wrong PMNC counter" |
| " %d\n", smp_processor_id(), cnt); |
| return -1; |
| } |
| |
| if (cnt == CCNT) |
| val = CNTENS_C; |
| else |
| val = (1 << (cnt - CNT0)); |
| |
| val &= CNTENS_MASK; |
| asm volatile("mcr p15, 0, %0, c9, c12, 1" : : "r" (val)); |
| |
| return cnt; |
| } |
| |
| static inline u32 armv7_pmnc_disable_counter(unsigned int cnt) |
| { |
| u32 val; |
| |
| if (cnt >= CNTMAX) { |
| printk(KERN_ERR "oprofile: CPU%u disabling wrong PMNC counter" |
| " %d\n", smp_processor_id(), cnt); |
| return -1; |
| } |
| |
| if (cnt == CCNT) |
| val = CNTENC_C; |
| else |
| val = (1 << (cnt - CNT0)); |
| |
| val &= CNTENC_MASK; |
| asm volatile("mcr p15, 0, %0, c9, c12, 2" : : "r" (val)); |
| |
| return cnt; |
| } |
| |
| static inline u32 armv7_pmnc_enable_intens(unsigned int cnt) |
| { |
| u32 val; |
| |
| if (cnt >= CNTMAX) { |
| printk(KERN_ERR "oprofile: CPU%u enabling wrong PMNC counter" |
| " interrupt enable %d\n", smp_processor_id(), cnt); |
| return -1; |
| } |
| |
| if (cnt == CCNT) |
| val = INTENS_C; |
| else |
| val = (1 << (cnt - CNT0)); |
| |
| val &= INTENS_MASK; |
| asm volatile("mcr p15, 0, %0, c9, c14, 1" : : "r" (val)); |
| |
| return cnt; |
| } |
| |
| static inline u32 armv7_pmnc_getreset_flags(void) |
| { |
| u32 val; |
| |
| /* Read */ |
| asm volatile("mrc p15, 0, %0, c9, c12, 3" : "=r" (val)); |
| |
| /* Write to clear flags */ |
| val &= FLAG_MASK; |
| asm volatile("mcr p15, 0, %0, c9, c12, 3" : : "r" (val)); |
| |
| return val; |
| } |
| |
| static inline int armv7_pmnc_select_counter(unsigned int cnt) |
| { |
| u32 val; |
| |
| if ((cnt == CCNT) || (cnt >= CNTMAX)) { |
| printk(KERN_ERR "oprofile: CPU%u selecting wrong PMNC counteri" |
| " %d\n", smp_processor_id(), cnt); |
| return -1; |
| } |
| |
| val = (cnt - CNT0) & SELECT_MASK; |
| asm volatile("mcr p15, 0, %0, c9, c12, 5" : : "r" (val)); |
| |
| return cnt; |
| } |
| |
| static inline void armv7_pmnc_write_evtsel(unsigned int cnt, u32 val) |
| { |
| if (armv7_pmnc_select_counter(cnt) == cnt) { |
| val &= EVTSEL_MASK; |
| asm volatile("mcr p15, 0, %0, c9, c13, 1" : : "r" (val)); |
| } |
| } |
| |
| static void armv7_pmnc_reset_counter(unsigned int cnt) |
| { |
| u32 cpu_cnt = CPU_COUNTER(smp_processor_id(), cnt); |
| u32 val = -(u32)counter_config[cpu_cnt].count; |
| |
| switch (cnt) { |
| case CCNT: |
| armv7_pmnc_disable_counter(cnt); |
| |
| asm volatile("mcr p15, 0, %0, c9, c13, 0" : : "r" (val)); |
| |
| if (cnt_en[cnt] != 0) |
| armv7_pmnc_enable_counter(cnt); |
| |
| break; |
| |
| case CNT0: |
| case CNT1: |
| case CNT2: |
| case CNT3: |
| armv7_pmnc_disable_counter(cnt); |
| |
| if (armv7_pmnc_select_counter(cnt) == cnt) |
| asm volatile("mcr p15, 0, %0, c9, c13, 2" : : "r" (val)); |
| |
| if (cnt_en[cnt] != 0) |
| armv7_pmnc_enable_counter(cnt); |
| |
| break; |
| |
| default: |
| printk(KERN_ERR "oprofile: CPU%u resetting wrong PMNC counter" |
| " %d\n", smp_processor_id(), cnt); |
| break; |
| } |
| } |
| |
| int armv7_setup_pmnc(void) |
| { |
| unsigned int cnt; |
| |
| if (armv7_pmnc_read() & PMNC_E) { |
| printk(KERN_ERR "oprofile: CPU%u PMNC still enabled when setup" |
| " new event counter.\n", smp_processor_id()); |
| return -EBUSY; |
| } |
| |
| /* |
| * Initialize & Reset PMNC: C bit, D bit and P bit. |
| * Note: Using a slower count for CCNT (D bit: divide by 64) results |
| * in a more stable system |
| */ |
| armv7_pmnc_write(PMNC_P | PMNC_C | PMNC_D); |
| |
| |
| for (cnt = CCNT; cnt < CNTMAX; cnt++) { |
| unsigned long event; |
| u32 cpu_cnt = CPU_COUNTER(smp_processor_id(), cnt); |
| |
| /* |
| * Disable counter |
| */ |
| armv7_pmnc_disable_counter(cnt); |
| cnt_en[cnt] = 0; |
| |
| if (!counter_config[cpu_cnt].enabled) |
| continue; |
| |
| event = counter_config[cpu_cnt].event & 255; |
| |
| /* |
| * Set event (if destined for PMNx counters) |
| * We don't need to set the event if it's a cycle count |
| */ |
| if (cnt != CCNT) |
| armv7_pmnc_write_evtsel(cnt, event); |
| |
| /* |
| * Enable interrupt for this counter |
| */ |
| armv7_pmnc_enable_intens(cnt); |
| |
| /* |
| * Reset counter |
| */ |
| armv7_pmnc_reset_counter(cnt); |
| |
| /* |
| * Enable counter |
| */ |
| armv7_pmnc_enable_counter(cnt); |
| cnt_en[cnt] = 1; |
| } |
| |
| return 0; |
| } |
| |
| static inline void armv7_start_pmnc(void) |
| { |
| armv7_pmnc_write(armv7_pmnc_read() | PMNC_E); |
| } |
| |
| static inline void armv7_stop_pmnc(void) |
| { |
| armv7_pmnc_write(armv7_pmnc_read() & ~PMNC_E); |
| } |
| |
| /* |
| * CPU counters' IRQ handler (one IRQ per CPU) |
| */ |
| static irqreturn_t armv7_pmnc_interrupt(int irq, void *arg) |
| { |
| struct pt_regs *regs = get_irq_regs(); |
| unsigned int cnt; |
| u32 flags; |
| |
| |
| /* |
| * Stop IRQ generation |
| */ |
| armv7_stop_pmnc(); |
| |
| /* |
| * Get and reset overflow status flags |
| */ |
| flags = armv7_pmnc_getreset_flags(); |
| |
| /* |
| * Cycle counter |
| */ |
| if (flags & FLAG_C) { |
| u32 cpu_cnt = CPU_COUNTER(smp_processor_id(), CCNT); |
| armv7_pmnc_reset_counter(CCNT); |
| oprofile_add_sample(regs, cpu_cnt); |
| } |
| |
| /* |
| * PMNC counters 0:3 |
| */ |
| for (cnt = CNT0; cnt < CNTMAX; cnt++) { |
| if (flags & (1 << (cnt - CNT0))) { |
| u32 cpu_cnt = CPU_COUNTER(smp_processor_id(), cnt); |
| armv7_pmnc_reset_counter(cnt); |
| oprofile_add_sample(regs, cpu_cnt); |
| } |
| } |
| |
| /* |
| * Allow IRQ generation |
| */ |
| armv7_start_pmnc(); |
| |
| return IRQ_HANDLED; |
| } |
| |
| int armv7_request_interrupts(int *irqs, int nr) |
| { |
| unsigned int i; |
| int ret = 0; |
| |
| for (i = 0; i < nr; i++) { |
| ret = request_irq(irqs[i], armv7_pmnc_interrupt, |
| IRQF_DISABLED, "CP15 PMNC", NULL); |
| if (ret != 0) { |
| printk(KERN_ERR "oprofile: unable to request IRQ%u" |
| " for ARMv7\n", |
| irqs[i]); |
| break; |
| } |
| } |
| |
| if (i != nr) |
| while (i-- != 0) |
| free_irq(irqs[i], NULL); |
| |
| return ret; |
| } |
| |
| void armv7_release_interrupts(int *irqs, int nr) |
| { |
| unsigned int i; |
| |
| for (i = 0; i < nr; i++) |
| free_irq(irqs[i], NULL); |
| } |
| |
| #ifdef DEBUG |
| static void armv7_pmnc_dump_regs(void) |
| { |
| u32 val; |
| unsigned int cnt; |
| |
| printk(KERN_INFO "PMNC registers dump:\n"); |
| |
| asm volatile("mrc p15, 0, %0, c9, c12, 0" : "=r" (val)); |
| printk(KERN_INFO "PMNC =0x%08x\n", val); |
| |
| asm volatile("mrc p15, 0, %0, c9, c12, 1" : "=r" (val)); |
| printk(KERN_INFO "CNTENS=0x%08x\n", val); |
| |
| asm volatile("mrc p15, 0, %0, c9, c14, 1" : "=r" (val)); |
| printk(KERN_INFO "INTENS=0x%08x\n", val); |
| |
| asm volatile("mrc p15, 0, %0, c9, c12, 3" : "=r" (val)); |
| printk(KERN_INFO "FLAGS =0x%08x\n", val); |
| |
| asm volatile("mrc p15, 0, %0, c9, c12, 5" : "=r" (val)); |
| printk(KERN_INFO "SELECT=0x%08x\n", val); |
| |
| asm volatile("mrc p15, 0, %0, c9, c13, 0" : "=r" (val)); |
| printk(KERN_INFO "CCNT =0x%08x\n", val); |
| |
| for (cnt = CNT0; cnt < CNTMAX; cnt++) { |
| armv7_pmnc_select_counter(cnt); |
| asm volatile("mrc p15, 0, %0, c9, c13, 2" : "=r" (val)); |
| printk(KERN_INFO "CNT[%d] count =0x%08x\n", cnt-CNT0, val); |
| asm volatile("mrc p15, 0, %0, c9, c13, 1" : "=r" (val)); |
| printk(KERN_INFO "CNT[%d] evtsel=0x%08x\n", cnt-CNT0, val); |
| } |
| } |
| #endif |
| |
| |
| static int irqs[] = { |
| #ifdef CONFIG_ARCH_OMAP3 |
| INT_34XX_BENCH_MPU_EMUL, |
| #endif |
| }; |
| |
| static void armv7_pmnc_stop(void) |
| { |
| #ifdef DEBUG |
| armv7_pmnc_dump_regs(); |
| #endif |
| armv7_stop_pmnc(); |
| armv7_release_interrupts(irqs, ARRAY_SIZE(irqs)); |
| } |
| |
| static int armv7_pmnc_start(void) |
| { |
| int ret; |
| |
| #ifdef DEBUG |
| armv7_pmnc_dump_regs(); |
| #endif |
| ret = armv7_request_interrupts(irqs, ARRAY_SIZE(irqs)); |
| if (ret >= 0) |
| armv7_start_pmnc(); |
| |
| return ret; |
| } |
| |
| static int armv7_detect_pmnc(void) |
| { |
| return 0; |
| } |
| |
| struct op_arm_model_spec op_armv7_spec = { |
| .init = armv7_detect_pmnc, |
| .num_counters = 5, |
| .setup_ctrs = armv7_setup_pmnc, |
| .start = armv7_pmnc_start, |
| .stop = armv7_pmnc_stop, |
| .name = "arm/armv7", |
| }; |