2005-04-16 15:20:36 -07:00
/**
* @ file oprof . c
*
* @ remark Copyright 2002 OProfile authors
* @ remark Read the file COPYING
*
* @ author John Levon < levon @ movementarian . org >
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/oprofile.h>
# include <linux/moduleparam.h>
2006-06-25 05:47:33 -07:00
# include <asm/mutex.h>
2005-04-16 15:20:36 -07:00
# include "oprof.h"
# include "event_buffer.h"
# include "cpu_buffer.h"
# include "buffer_sync.h"
# include "oprofile_stats.h"
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
struct oprofile_operations oprofile_ops ;
unsigned long oprofile_started ;
unsigned long backtrace_depth ;
2008-09-24 11:08:52 +02:00
static unsigned long is_setup ;
static DEFINE_MUTEX ( start_mutex ) ;
2005-04-16 15:20:36 -07:00
/* timer
0 - use performance monitoring hardware if available
1 - use the timer int mechanism regardless
*/
static int timer = 0 ;
int oprofile_setup ( void )
{
int err ;
2008-09-05 17:12:36 +02:00
2006-06-25 05:47:33 -07:00
mutex_lock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
if ( ( err = alloc_cpu_buffers ( ) ) )
goto out ;
if ( ( err = alloc_event_buffer ( ) ) )
goto out1 ;
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
if ( oprofile_ops . setup & & ( err = oprofile_ops . setup ( ) ) )
goto out2 ;
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
/* Note even though this starts part of the
* profiling overhead , it ' s necessary to prevent
* us missing task deaths and eventually oopsing
* when trying to process the event buffer .
*/
2007-07-20 21:39:53 +02:00
if ( oprofile_ops . sync_start ) {
int sync_ret = oprofile_ops . sync_start ( ) ;
switch ( sync_ret ) {
case 0 :
goto post_sync ;
case 1 :
goto do_generic ;
case - 1 :
goto out3 ;
default :
goto out3 ;
}
}
do_generic :
2005-04-16 15:20:36 -07:00
if ( ( err = sync_start ( ) ) )
goto out3 ;
2007-07-20 21:39:53 +02:00
post_sync :
2005-04-16 15:20:36 -07:00
is_setup = 1 ;
2006-06-25 05:47:33 -07:00
mutex_unlock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
out3 :
if ( oprofile_ops . shutdown )
oprofile_ops . shutdown ( ) ;
out2 :
free_event_buffer ( ) ;
out1 :
free_cpu_buffers ( ) ;
out :
2006-06-25 05:47:33 -07:00
mutex_unlock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
/* Actually start profiling (echo 1>/dev/oprofile/enable) */
int oprofile_start ( void )
{
int err = - EINVAL ;
2008-09-05 17:12:36 +02:00
2006-06-25 05:47:33 -07:00
mutex_lock ( & start_mutex ) ;
2008-10-16 15:01:40 +02:00
2005-04-16 15:20:36 -07:00
if ( ! is_setup )
goto out ;
2008-09-05 17:12:36 +02:00
err = 0 ;
2005-04-16 15:20:36 -07:00
if ( oprofile_started )
goto out ;
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
oprofile_reset_stats ( ) ;
if ( ( err = oprofile_ops . start ( ) ) )
goto out ;
oprofile_started = 1 ;
out :
2006-06-25 05:47:33 -07:00
mutex_unlock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
/* echo 0>/dev/oprofile/enable */
void oprofile_stop ( void )
{
2006-06-25 05:47:33 -07:00
mutex_lock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
if ( ! oprofile_started )
goto out ;
oprofile_ops . stop ( ) ;
oprofile_started = 0 ;
/* wake up the daemon to read what remains */
wake_up_buffer_waiter ( ) ;
out :
2006-06-25 05:47:33 -07:00
mutex_unlock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
}
void oprofile_shutdown ( void )
{
2006-06-25 05:47:33 -07:00
mutex_lock ( & start_mutex ) ;
2007-07-20 21:39:53 +02:00
if ( oprofile_ops . sync_stop ) {
int sync_ret = oprofile_ops . sync_stop ( ) ;
switch ( sync_ret ) {
case 0 :
goto post_sync ;
case 1 :
goto do_generic ;
default :
goto post_sync ;
}
}
do_generic :
2005-04-16 15:20:36 -07:00
sync_stop ( ) ;
2007-07-20 21:39:53 +02:00
post_sync :
2005-04-16 15:20:36 -07:00
if ( oprofile_ops . shutdown )
oprofile_ops . shutdown ( ) ;
is_setup = 0 ;
free_event_buffer ( ) ;
free_cpu_buffers ( ) ;
2006-06-25 05:47:33 -07:00
mutex_unlock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
}
int oprofile_set_backtrace ( unsigned long val )
{
int err = 0 ;
2006-06-25 05:47:33 -07:00
mutex_lock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
if ( oprofile_started ) {
err = - EBUSY ;
goto out ;
}
if ( ! oprofile_ops . backtrace ) {
err = - EINVAL ;
goto out ;
}
backtrace_depth = val ;
out :
2006-06-25 05:47:33 -07:00
mutex_unlock ( & start_mutex ) ;
2005-04-16 15:20:36 -07:00
return err ;
}
static int __init oprofile_init ( void )
{
int err ;
err = oprofile_arch_init ( & oprofile_ops ) ;
if ( err < 0 | | timer ) {
printk ( KERN_INFO " oprofile: using timer interrupt. \n " ) ;
oprofile_timer_init ( & oprofile_ops ) ;
}
err = oprofilefs_register ( ) ;
if ( err )
oprofile_arch_exit ( ) ;
return err ;
}
static void __exit oprofile_exit ( void )
{
oprofilefs_unregister ( ) ;
oprofile_arch_exit ( ) ;
}
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
module_init ( oprofile_init ) ;
module_exit ( oprofile_exit ) ;
module_param_named ( timer , timer , int , 0644 ) ;
MODULE_PARM_DESC ( timer , " force use of timer interrupt " ) ;
2008-09-05 17:12:36 +02:00
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " John Levon <levon@movementarian.org> " ) ;
MODULE_DESCRIPTION ( " OProfile system profiler " ) ;