- 论坛徽章:
- 0
|
- #include <linux/module.h>
- #include <linux/moduleparam.h>
- #include <linux/init.h>
- #include <linux/time.h>
- #include <linux/timer.h>
- #include <linux/kernel.h>
- #include <linux/proc_fs.h>
- #include <linux/types.h>
- #include <linux/spinlock.h>
- #include <linux/interrupt.h>
- #include <asm/hardirq.h>
- /*
- * This module is a silly one: it only embeds short code fragments
- * that show how time delays can be handled in the kernel.
- */
- int delay = HZ; /* the default delay, expressed in jiffies */
- module_param(delay, int, 0);
- MODULE_AUTHOR("Alessandro Rubini");
- MODULE_LICENSE("Dual BSD/GPL");
- /* This data structure used as "data" for the timer and tasklet functions */
- struct jit_data {
- struct timer_list timer;
- struct tasklet_struct tlet;
- int hi; /* tasklet or tasklet_hi */
- wait_queue_head_t wait;
- unsigned long prevjiffies;
- unsigned char *buf;
- int loops;
- };
- #define JIT_ASYNC_LOOPS 5
- void jit_tasklet_fn(unsigned long arg)
- {
- struct jit_data *data = (struct jit_data *)arg;
- unsigned long j = jiffies;
- data->buf += sprintf(data->buf, "%9li %3li %i %6i %i %s\n",
- j, j - data->prevjiffies, in_interrupt() ? 1 : 0,
- current->pid, smp_processor_id(), current->comm);
- if (--data->loops) {
- data->prevjiffies = j;
- if (data->hi)
- tasklet_hi_schedule(&data->tlet);
- else
- tasklet_schedule(&data->tlet);
- } else {
- wake_up_interruptible(&data->wait);
- }
- }
- /* the /proc function: allocate everything to allow concurrency */
- int jit_tasklet(char *buf, char **start, off_t offset,
- int len, int *eof, void *arg)
- {
- struct jit_data *data;
- char *buf2 = buf;
- unsigned long j = jiffies;
- long hi = (long)arg;
- data = kmalloc(sizeof(*data), GFP_KERNEL);
- if (!data)
- return -ENOMEM;
- init_waitqueue_head (&data->wait);
- /* write the first lines in the buffer */
- buf2 += sprintf(buf2, " time delta inirq pid cpu command\n");
- buf2 += sprintf(buf2, "%9li %3li %i %6i %i %s\n",
- j, 0L, in_interrupt() ? 1 : 0,
- current->pid, smp_processor_id(), current->comm);
- /* fill the data for our tasklet function */
- data->prevjiffies = j;
- data->buf = buf2;
- data->loops = JIT_ASYNC_LOOPS;
-
- /* register the tasklet */
- tasklet_init(&data->tlet, jit_tasklet_fn, (unsigned long)data);
- data->hi = hi;
- if (hi)
- tasklet_hi_schedule(&data->tlet);
- else
- tasklet_schedule(&data->tlet);
- /* wait for the buffer to fill */
- wait_event_interruptible(data->wait, !data->loops);
- if (signal_pending(current))
- return -ERESTARTSYS;
- buf2 = data->buf;
- kfree(data);
- *eof = 1;
- return buf2 - buf;
- }
- int __init jit_init(void)
- {
- create_proc_read_entry("jitasklet", 0, NULL, jit_tasklet, NULL);
- create_proc_read_entry("jitasklethi", 0, NULL, jit_tasklet, (void *)1);
- return 0; /* success */
- }
- void __exit jit_cleanup(void)
- {
- remove_proc_entry("jitasklet", NULL);
- remove_proc_entry("jitasklethi", NULL);
- }
- module_init(jit_init);
- module_exit(jit_cleanup);
复制代码 |
|