2019-11-20 22:08:39 +03:00
// SPDX-License-Identifier: GPL-2.0-only
# include <linux/module.h>
# include <linux/kthread.h>
# include <linux/trace.h>
# include <linux/trace_events.h>
# include <linux/timer.h>
# include <linux/err.h>
# include <linux/jiffies.h>
2020-06-10 04:12:44 +03:00
# include <linux/workqueue.h>
2019-11-20 22:08:39 +03:00
/*
* Any file that uses trace points , must include the header .
* But only one file , must include the header by defining
* CREATE_TRACE_POINTS first . This will make the C code that
* creates the handles for the trace points .
*/
# define CREATE_TRACE_POINTS
# include "sample-trace-array.h"
struct trace_array * tr ;
static void mytimer_handler ( struct timer_list * unused ) ;
static struct task_struct * simple_tsk ;
2020-06-10 04:12:44 +03:00
static void trace_work_fn ( struct work_struct * work )
{
/*
* Disable tracing for event " sample_event " .
*/
trace_array_set_clr_event ( tr , " sample-subsystem " , " sample_event " ,
false ) ;
}
static DECLARE_WORK ( trace_work , trace_work_fn ) ;
2019-11-20 22:08:39 +03:00
/*
* mytimer : Timer setup to disable tracing for event " sample_event " . This
* timer is only for the purposes of the sample module to demonstrate access of
* Ftrace instances from within kernel .
*/
static DEFINE_TIMER ( mytimer , mytimer_handler ) ;
static void mytimer_handler ( struct timer_list * unused )
{
2020-06-10 04:12:44 +03:00
schedule_work ( & trace_work ) ;
2019-11-20 22:08:39 +03:00
}
static void simple_thread_func ( int count )
{
set_current_state ( TASK_INTERRUPTIBLE ) ;
schedule_timeout ( HZ ) ;
/*
* Printing count value using trace_array_printk ( ) - trace_printk ( )
* equivalent for the instance buffers .
*/
trace_array_printk ( tr , _THIS_IP_ , " trace_array_printk: count=%d \n " ,
count ) ;
/*
* Tracepoint for event " sample_event " . This will print the
* current value of count and current jiffies .
*/
trace_sample_event ( count , jiffies ) ;
}
static int simple_thread ( void * arg )
{
int count = 0 ;
unsigned long delay = msecs_to_jiffies ( 5000 ) ;
/*
* Enable tracing for " sample_event " .
*/
trace_array_set_clr_event ( tr , " sample-subsystem " , " sample_event " , true ) ;
/*
* Adding timer - mytimer . This timer will disable tracing after
* delay seconds .
*
*/
add_timer ( & mytimer ) ;
mod_timer ( & mytimer , jiffies + delay ) ;
while ( ! kthread_should_stop ( ) )
simple_thread_func ( count + + ) ;
del_timer ( & mytimer ) ;
2020-06-10 04:12:44 +03:00
cancel_work_sync ( & trace_work ) ;
2019-11-20 22:08:39 +03:00
/*
* trace_array_put ( ) decrements the reference counter associated with
* the trace array - " tr " . We are done using the trace array , hence
* decrement the reference counter so that it can be destroyed using
* trace_array_destroy ( ) .
*/
trace_array_put ( tr ) ;
return 0 ;
}
static int __init sample_trace_array_init ( void )
{
/*
* Return a pointer to the trace array with name " sample-instance " if it
* exists , else create a new trace array .
*
* NOTE : This function increments the reference counter
* associated with the trace array - " tr " .
*/
2023-12-13 17:37:01 +03:00
tr = trace_array_get_by_name ( " sample-instance " , " sched,timer,kprobes " ) ;
2019-11-20 22:08:39 +03:00
if ( ! tr )
return - 1 ;
/*
* If context specific per - cpu buffers havent already been allocated .
*/
trace_printk_init_buffers ( ) ;
simple_tsk = kthread_run ( simple_thread , NULL , " sample-instance " ) ;
2020-06-09 16:52:00 +03:00
if ( IS_ERR ( simple_tsk ) ) {
trace_array_put ( tr ) ;
trace_array_destroy ( tr ) ;
2019-11-20 22:08:39 +03:00
return - 1 ;
2020-06-09 16:52:00 +03:00
}
2019-11-20 22:08:39 +03:00
return 0 ;
}
static void __exit sample_trace_array_exit ( void )
{
kthread_stop ( simple_tsk ) ;
/*
* We are unloading our module and no longer require the trace array .
* Remove / destroy " tr " using trace_array_destroy ( )
*/
trace_array_destroy ( tr ) ;
}
module_init ( sample_trace_array_init ) ;
module_exit ( sample_trace_array_exit ) ;
MODULE_AUTHOR ( " Divya Indi " ) ;
MODULE_DESCRIPTION ( " Sample module for kernel access to Ftrace instances " ) ;
MODULE_LICENSE ( " GPL " ) ;