2005-04-17 02:20:36 +04: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>
2009-07-08 15:49:38 +04:00
# include <linux/workqueue.h>
# include <linux/time.h>
2006-06-25 16:47:33 +04:00
# include <asm/mutex.h>
2005-04-17 02:20:36 +04:00
# include "oprof.h"
# include "event_buffer.h"
# include "cpu_buffer.h"
# include "buffer_sync.h"
# include "oprofile_stats.h"
2008-09-05 19:12:36 +04:00
2005-04-17 02:20:36 +04:00
struct oprofile_operations oprofile_ops ;
unsigned long oprofile_started ;
2008-12-16 18:19:54 +03:00
unsigned long oprofile_backtrace_depth ;
2008-09-24 13:08:52 +04:00
static unsigned long is_setup ;
static DEFINE_MUTEX ( start_mutex ) ;
2005-04-17 02:20:36 +04:00
2009-07-08 15:49:38 +04:00
# ifdef CONFIG_OPROFILE_EVENT_MULTIPLEX
static void switch_worker ( struct work_struct * work ) ;
static DECLARE_DELAYED_WORK ( switch_work , switch_worker ) ;
# endif
2005-04-17 02:20:36 +04: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 19:12:36 +04:00
2006-06-25 16:47:33 +04:00
mutex_lock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( ( err = alloc_cpu_buffers ( ) ) )
goto out ;
if ( ( err = alloc_event_buffer ( ) ) )
goto out1 ;
2008-09-05 19:12:36 +04:00
2005-04-17 02:20:36 +04:00
if ( oprofile_ops . setup & & ( err = oprofile_ops . setup ( ) ) )
goto out2 ;
2008-09-05 19:12:36 +04:00
2005-04-17 02:20:36 +04: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 23:39:53 +04: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-17 02:20:36 +04:00
if ( ( err = sync_start ( ) ) )
goto out3 ;
2007-07-20 23:39:53 +04:00
post_sync :
2005-04-17 02:20:36 +04:00
is_setup = 1 ;
2006-06-25 16:47:33 +04:00
mutex_unlock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
2008-09-05 19:12:36 +04:00
2005-04-17 02:20:36 +04:00
out3 :
if ( oprofile_ops . shutdown )
oprofile_ops . shutdown ( ) ;
out2 :
free_event_buffer ( ) ;
out1 :
free_cpu_buffers ( ) ;
out :
2006-06-25 16:47:33 +04:00
mutex_unlock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
return err ;
}
2009-07-08 15:49:38 +04:00
# ifdef CONFIG_OPROFILE_EVENT_MULTIPLEX
static void start_switch_worker ( void )
{
2009-07-15 17:19:29 +04:00
schedule_delayed_work ( & switch_work , oprofile_time_slice ) ;
2009-07-08 15:49:38 +04:00
}
static void switch_worker ( struct work_struct * work )
{
if ( ! oprofile_ops . switch_events ( ) )
start_switch_worker ( ) ;
}
# endif
2005-04-17 02:20:36 +04:00
/* Actually start profiling (echo 1>/dev/oprofile/enable) */
int oprofile_start ( void )
{
int err = - EINVAL ;
2008-09-05 19:12:36 +04:00
2006-06-25 16:47:33 +04:00
mutex_lock ( & start_mutex ) ;
2008-10-16 17:01:40 +04:00
2005-04-17 02:20:36 +04:00
if ( ! is_setup )
goto out ;
2008-09-05 19:12:36 +04:00
err = 0 ;
2005-04-17 02:20:36 +04:00
if ( oprofile_started )
goto out ;
2008-09-05 19:12:36 +04:00
2005-04-17 02:20:36 +04:00
oprofile_reset_stats ( ) ;
if ( ( err = oprofile_ops . start ( ) ) )
goto out ;
2009-07-08 15:49:38 +04:00
# ifdef CONFIG_OPROFILE_EVENT_MULTIPLEX
if ( oprofile_ops . switch_events )
start_switch_worker ( ) ;
# endif
2005-04-17 02:20:36 +04:00
oprofile_started = 1 ;
out :
2006-06-25 16:47:33 +04:00
mutex_unlock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
return err ;
}
2008-09-05 19:12:36 +04:00
2005-04-17 02:20:36 +04:00
/* echo 0>/dev/oprofile/enable */
void oprofile_stop ( void )
{
2006-06-25 16:47:33 +04:00
mutex_lock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( ! oprofile_started )
goto out ;
oprofile_ops . stop ( ) ;
oprofile_started = 0 ;
2009-07-08 15:49:38 +04:00
# ifdef CONFIG_OPROFILE_EVENT_MULTIPLEX
cancel_delayed_work_sync ( & switch_work ) ;
# endif
2005-04-17 02:20:36 +04:00
/* wake up the daemon to read what remains */
wake_up_buffer_waiter ( ) ;
out :
2006-06-25 16:47:33 +04:00
mutex_unlock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
}
void oprofile_shutdown ( void )
{
2006-06-25 16:47:33 +04:00
mutex_lock ( & start_mutex ) ;
2007-07-20 23:39:53 +04: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-17 02:20:36 +04:00
sync_stop ( ) ;
2007-07-20 23:39:53 +04:00
post_sync :
2005-04-17 02:20:36 +04:00
if ( oprofile_ops . shutdown )
oprofile_ops . shutdown ( ) ;
is_setup = 0 ;
free_event_buffer ( ) ;
free_cpu_buffers ( ) ;
2006-06-25 16:47:33 +04:00
mutex_unlock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
}
2009-07-08 15:49:38 +04:00
# ifdef CONFIG_OPROFILE_EVENT_MULTIPLEX
/* User inputs in ms, converts to jiffies */
int oprofile_set_timeout ( unsigned long val_msec )
{
int err = 0 ;
2009-07-15 17:44:18 +04:00
unsigned long time_slice ;
2009-07-08 15:49:38 +04:00
mutex_lock ( & start_mutex ) ;
if ( oprofile_started ) {
err = - EBUSY ;
goto out ;
}
if ( ! oprofile_ops . switch_events ) {
err = - EINVAL ;
goto out ;
}
2009-07-15 17:44:18 +04:00
time_slice = msecs_to_jiffies ( val_msec ) ;
if ( time_slice = = MAX_JIFFY_OFFSET ) {
err = - EINVAL ;
goto out ;
}
2009-07-15 17:19:29 +04:00
oprofile_time_slice = time_slice ;
2009-07-08 15:49:38 +04:00
out :
mutex_unlock ( & start_mutex ) ;
return err ;
}
# endif
2005-04-17 02:20:36 +04:00
int oprofile_set_backtrace ( unsigned long val )
{
int err = 0 ;
2006-06-25 16:47:33 +04:00
mutex_lock ( & start_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( oprofile_started ) {
err = - EBUSY ;
goto out ;
}
if ( ! oprofile_ops . backtrace ) {
err = - EINVAL ;
goto out ;
}
2008-12-16 18:19:54 +03:00
oprofile_backtrace_depth = val ;
2005-04-17 02:20:36 +04:00
out :
2006-06-25 16:47:33 +04:00
mutex_unlock ( & start_mutex ) ;
2005-04-17 02:20:36 +04: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 ( ) ;
2009-01-22 16:14:14 +03:00
if ( err )
2005-04-17 02:20:36 +04:00
oprofile_arch_exit ( ) ;
return err ;
}
static void __exit oprofile_exit ( void )
{
oprofilefs_unregister ( ) ;
oprofile_arch_exit ( ) ;
}
2008-09-05 19:12:36 +04:00
2005-04-17 02:20:36 +04: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 19:12:36 +04:00
2005-04-17 02:20:36 +04:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " John Levon <levon@movementarian.org> " ) ;
MODULE_DESCRIPTION ( " OProfile system profiler " ) ;