2005-04-17 02:20:36 +04:00
/*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*
2005-02-07 05:54:29 +03:00
* Copyright ( C ) 2004 , 2005 Ralf Baechle
* Copyright ( C ) 2005 MIPS Technologies , Inc .
2005-04-17 02:20:36 +04:00
*/
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/oprofile.h>
# include <linux/smp.h>
# include <asm/cpu-info.h>
# include "op_impl.h"
2006-05-23 11:42:38 +04:00
extern struct op_mips_model op_model_mipsxx_ops __attribute__ ( ( weak ) ) ;
extern struct op_mips_model op_model_rm9000_ops __attribute__ ( ( weak ) ) ;
2005-04-17 02:20:36 +04:00
static struct op_mips_model * model ;
static struct op_counter_config ctr [ 20 ] ;
static int op_mips_setup ( void )
{
/* Pre-compute the values to stuff in the hardware registers. */
model - > reg_setup ( ctr ) ;
/* Configure the registers on all cpus. */
2006-04-24 13:50:02 +04:00
on_each_cpu ( model - > cpu_setup , NULL , 0 , 1 ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int op_mips_create_files ( struct super_block * sb , struct dentry * root )
{
int i ;
for ( i = 0 ; i < model - > num_counters ; + + i ) {
struct dentry * dir ;
2006-06-26 11:24:34 +04:00
char buf [ 4 ] ;
2005-04-17 02:20:36 +04:00
snprintf ( buf , sizeof buf , " %d " , i ) ;
dir = oprofilefs_mkdir ( sb , root , buf ) ;
oprofilefs_create_ulong ( sb , dir , " enabled " , & ctr [ i ] . enabled ) ;
oprofilefs_create_ulong ( sb , dir , " event " , & ctr [ i ] . event ) ;
oprofilefs_create_ulong ( sb , dir , " count " , & ctr [ i ] . count ) ;
oprofilefs_create_ulong ( sb , dir , " kernel " , & ctr [ i ] . kernel ) ;
oprofilefs_create_ulong ( sb , dir , " user " , & ctr [ i ] . user ) ;
oprofilefs_create_ulong ( sb , dir , " exl " , & ctr [ i ] . exl ) ;
2005-02-07 05:54:29 +03:00
/* Dummy. */
2005-04-17 02:20:36 +04:00
oprofilefs_create_ulong ( sb , dir , " unit_mask " , & ctr [ i ] . unit_mask ) ;
}
return 0 ;
}
static int op_mips_start ( void )
{
on_each_cpu ( model - > cpu_start , NULL , 0 , 1 ) ;
return 0 ;
}
static void op_mips_stop ( void )
{
/* Disable performance monitoring for all counters. */
on_each_cpu ( model - > cpu_stop , NULL , 0 , 1 ) ;
}
2005-02-07 05:54:29 +03:00
int __init oprofile_arch_init ( struct oprofile_operations * ops )
2005-04-17 02:20:36 +04:00
{
struct op_mips_model * lmodel = NULL ;
2005-02-07 05:54:29 +03:00
int res ;
2005-04-17 02:20:36 +04:00
2007-10-12 02:46:15 +04:00
switch ( current_cpu_type ( ) ) {
2005-12-09 15:42:13 +03:00
case CPU_5KC :
case CPU_20KC :
2005-04-17 02:20:36 +04:00
case CPU_24K :
2005-12-09 15:42:13 +03:00
case CPU_25KF :
2006-02-01 20:54:30 +03:00
case CPU_34K :
2006-05-02 17:08:46 +04:00
case CPU_74K :
2006-01-17 23:06:32 +03:00
case CPU_SB1 :
case CPU_SB1A :
2006-10-23 03:44:02 +04:00
case CPU_R10000 :
case CPU_R12000 :
case CPU_R14000 :
2006-05-23 11:42:38 +04:00
lmodel = & op_model_mipsxx_ops ;
2005-04-17 02:20:36 +04:00
break ;
case CPU_RM9000 :
2006-05-23 11:42:38 +04:00
lmodel = & op_model_rm9000_ops ;
2005-04-17 02:20:36 +04:00
break ;
} ;
if ( ! lmodel )
2005-02-07 05:54:29 +03:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2005-02-07 05:54:29 +03:00
res = lmodel - > init ( ) ;
if ( res )
return res ;
2005-04-17 02:20:36 +04:00
model = lmodel ;
2005-02-07 05:54:29 +03:00
ops - > create_files = op_mips_create_files ;
ops - > setup = op_mips_setup ;
//ops->shutdown = op_mips_shutdown;
ops - > start = op_mips_start ;
ops - > stop = op_mips_stop ;
ops - > cpu_type = lmodel - > cpu_type ;
2005-04-17 02:20:36 +04:00
printk ( KERN_INFO " oprofile: using %s performance monitoring. \n " ,
lmodel - > cpu_type ) ;
2005-02-07 05:54:29 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
void oprofile_arch_exit ( void )
{
2006-04-24 13:48:54 +04:00
if ( model )
model - > exit ( ) ;
2005-04-17 02:20:36 +04:00
}