2005-04-16 15:20:36 -07:00
/*
2005-10-19 14:53:32 +10:00
* arch / powerpc / kernel / pmc . c
2005-04-16 15:20:36 -07:00
*
* Copyright ( C ) 2004 David Gibson , IBM Corporation .
2005-10-19 14:53:32 +10:00
* Includes code formerly from arch / ppc / kernel / perfmon . c :
* Author : Andy Fleming
* Copyright ( c ) 2004 Freescale Semiconductor , Inc
2005-04-16 15:20:36 -07:00
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# include <linux/config.h>
# include <linux/errno.h>
# include <linux/spinlock.h>
# include <linux/module.h>
# include <asm/processor.h>
# include <asm/pmc.h>
2005-10-19 14:53:32 +10:00
# if defined(CONFIG_FSL_BOOKE) && !defined(CONFIG_E200)
static void dummy_perf ( struct pt_regs * regs )
{
unsigned int pmgc0 = mfpmr ( PMRN_PMGC0 ) ;
pmgc0 & = ~ PMGC0_PMIE ;
mtpmr ( PMRN_PMGC0 , pmgc0 ) ;
}
# elif defined(CONFIG_PPC64) || defined(CONFIG_6xx)
# ifndef MMCR0_PMAO
# define MMCR0_PMAO 0
# endif
2005-04-16 15:20:36 -07:00
/* Ensure exceptions are disabled */
static void dummy_perf ( struct pt_regs * regs )
{
unsigned int mmcr0 = mfspr ( SPRN_MMCR0 ) ;
mmcr0 & = ~ ( MMCR0_PMXE | MMCR0_PMAO ) ;
mtspr ( SPRN_MMCR0 , mmcr0 ) ;
}
2005-10-19 14:53:32 +10:00
# else
static void dummy_perf ( struct pt_regs * regs )
{
}
# endif
2005-04-16 15:20:36 -07:00
2005-09-09 13:10:41 -07:00
static DEFINE_SPINLOCK ( pmc_owner_lock ) ;
2005-04-16 15:20:36 -07:00
static void * pmc_owner_caller ; /* mostly for debugging */
perf_irq_t perf_irq = dummy_perf ;
int reserve_pmc_hardware ( perf_irq_t new_perf_irq )
{
int err = 0 ;
spin_lock ( & pmc_owner_lock ) ;
if ( pmc_owner_caller ) {
printk ( KERN_WARNING " reserve_pmc_hardware: "
" PMC hardware busy (reserved by caller %p) \n " ,
pmc_owner_caller ) ;
err = - EBUSY ;
goto out ;
}
pmc_owner_caller = __builtin_return_address ( 0 ) ;
perf_irq = new_perf_irq ? : dummy_perf ;
out :
spin_unlock ( & pmc_owner_lock ) ;
return err ;
}
EXPORT_SYMBOL_GPL ( reserve_pmc_hardware ) ;
void release_pmc_hardware ( void )
{
spin_lock ( & pmc_owner_lock ) ;
WARN_ON ( ! pmc_owner_caller ) ;
pmc_owner_caller = NULL ;
perf_irq = dummy_perf ;
spin_unlock ( & pmc_owner_lock ) ;
}
EXPORT_SYMBOL_GPL ( release_pmc_hardware ) ;
2005-08-09 11:13:36 +10:00
2005-10-19 14:53:32 +10:00
# ifdef CONFIG_PPC64
2005-08-09 11:13:36 +10:00
void power4_enable_pmcs ( void )
{
unsigned long hid0 ;
2005-10-10 14:01:07 +10:00
hid0 = mfspr ( SPRN_HID0 ) ;
2005-08-09 11:13:36 +10:00
hid0 | = 1UL < < ( 63 - 20 ) ;
/* POWER4 requires the following sequence */
asm volatile (
" sync \n "
" mtspr %1, %0 \n "
" mfspr %0, %1 \n "
" mfspr %0, %1 \n "
" mfspr %0, %1 \n "
" mfspr %0, %1 \n "
" mfspr %0, %1 \n "
" mfspr %0, %1 \n "
2005-10-10 14:01:07 +10:00
" isync " : " =&r " ( hid0 ) : " i " ( SPRN_HID0 ) , " 0 " ( hid0 ) :
2005-08-09 11:13:36 +10:00
" memory " ) ;
}
2005-10-19 14:53:32 +10:00
# endif /* CONFIG_PPC64 */