• Linux内核之tasklet机制


    tasklet机制用于queue up work等到未来某时执行。tasklet可以并行执行,但同一个tasklet在某一时刻只能运行于某一CPU上。为了cache优化,每个tasklet只会在调度它的CPU上运行。

    tasklet就好比一个既没有stack也没有context的小巧thread,它会快速执行完毕

    概要

    使用tasklet需要注意一些原则:

    1. tasklets是原子操作的,所以不能调用sleep()等阻塞函数以及mutexes(互斥量),semaphores(信号量)等,不过可以使用自旋锁(spinlock)
    2. A tasklet only runs on the same core (CPU) that schedules it
    3. Different tasklets can be running in parallel. But at the same time, a tasklet cannot be called concurrently with itself, as it runs on one CPU only
    4. tasklets的调度机制必须是non-preemptive调度器,一个接一个的执行,不能抢占。所以可以用normal 和 high两种优先级调度它

    tasklet structure

    Create tasklet

    include/linux/interrupt.h头文件中定义

    /* Tasklets --- multithreaded analogue of BHs.
    
       This API is deprecated. Please consider using threaded IRQs instead:
       https://lore.kernel.org/lkml/20200716081538.2sivhkj4hcyrusem@linutronix.de
    
       Main feature differing them of generic softirqs: tasklet
       is running only on one CPU simultaneously.
    
       Main feature differing them of BHs: different tasklets
       may be run simultaneously on different CPUs.
    
       Properties:
       * If tasklet_schedule() is called, then tasklet is guaranteed
         to be executed on some cpu at least once after this.
       * If the tasklet is already scheduled, but its execution is still not
         started, it will be executed only once.
       * If this tasklet is already running on another CPU (or schedule is called
         from tasklet itself), it is rescheduled for later.
       * Tasklet is strictly serialized wrt itself, but not
         wrt another tasklets. If client needs some intertask synchronization,
         he makes it with spinlocks.
     */
    
    struct tasklet_struct
    {
    	struct tasklet_struct *next;
    	unsigned long state;
    	atomic_t count;
    	bool use_callback;
    	union {
    		void (*func)(unsigned long data);
    		void (*callback)(struct tasklet_struct *t);
    	};
    	unsigned long data;
    };
    
    #define DECLARE_TASKLET(name, _callback)		\
    struct tasklet_struct name = {				\
    	.count = ATOMIC_INIT(0),			\
    	.callback = _callback,				\
    	.use_callback = true,				\
    }
    
    #define DECLARE_TASKLET_DISABLED(name, _callback)	\
    struct tasklet_struct name = {				\
    	.count = ATOMIC_INIT(1),			\
    	.callback = _callback,				\
    	.use_callback = true,				\
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49

    DECLARE_TASKLET()宏创建了一个tasklet结构体,并赋值其enable(count值为0表示enable,count值为非0,表示tasklet是disabled)。如果使用DECLARE_TASKLET_DISABLE()宏创建tasklet,该tasklet可以被调度,但其实disable的,需要enable之后才能实际运行。

    Eanble/Disable
    static inline void tasklet_disable(struct tasklet_struct *t)
    {
    	tasklet_disable_nosync(t);
    	tasklet_unlock_wait(t);
    	smp_mb();
    }
    
    static inline void tasklet_enable(struct tasklet_struct *t)
    {
    	smp_mb__before_atomic();
    	atomic_dec(&t->count);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    如果tasklet是disabled状态的,它是可以被加入queue然后被schedule的,但不会在CPU上运行,直至它被enable之后才会实际运行。如果某个tasklet被disable了n次, 那么对应的也需要enable它n次(count会计数),它才能真正被enabled

    Schedule tasklet

    当调度tasklet时,tasklet会被放入队列,一共是两种队列,normal和high两种优先级,单链表结构组织,每个CPU都有它自己的tasklet队列。

    • normal priority
    • high prioriy
    tasklet_schedule

    调度tasklet使用normal优先级

    include/linux/interrupt.h

    enum
    {
    	TASKLET_STATE_SCHED,	/* Tasklet is scheduled for execution */
    	TASKLET_STATE_RUN	/* Tasklet is running (SMP only) */
    };
    
    static inline void tasklet_schedule(struct tasklet_struct *t)
    {
    	if (!test_and_set_bit(TASKLET_STATE_SCHED, &t->state))
    		__tasklet_schedule(t);
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    kernel/softirq.c

    /*
     * This function must run with irqs disabled!
     */
    inline void raise_softirq_irqoff(unsigned int nr)
    {
    	__raise_softirq_irqoff(nr);
    
    	/*
    	 * If we're in an interrupt or softirq, we're done
    	 * (this also catches softirq-disabled code). We will
    	 * actually run the softirq once we return from
    	 * the irq or softirq.
    	 *
    	 * Otherwise we wake up ksoftirqd to make sure we
    	 * schedule the softirq soon.
    	 */
    	if (!in_interrupt() && should_wake_ksoftirqd())  /*会判断是否在中断环境中*/
    		wakeup_softirqd();
    }
    
    static void __tasklet_schedule_common(struct tasklet_struct *t,
    				      struct tasklet_head __percpu *headp,
    				      unsigned int softirq_nr)
    {
    	struct tasklet_head *head;
    	unsigned long flags;
    
    	local_irq_save(flags);
    	head = this_cpu_ptr(headp);
    	t->next = NULL;
    	*head->tail = t;
    	head->tail = &(t->next);
    	raise_softirq_irqoff(softirq_nr);
    	local_irq_restore(flags);
    }
    
    void __tasklet_schedule(struct tasklet_struct *t)
    {
    	__tasklet_schedule_common(t, &tasklet_vec,
    				  TASKLET_SOFTIRQ);
    }
    EXPORT_SYMBOL(__tasklet_schedule);
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    tasklet_schedule
    __tasklet_schedule
    __tasklet_schedule_common
    raise_softirq_irqoff

    调度tasklet最后是通过TASKLET_SOFTIRQ软中断去触发ksoftirqd内核线程来完成调度。

    tasklet_hi_schedule

    调度tasklet使用high优先级

    include/linux/interrupt.h

    static inline void tasklet_hi_schedule(struct tasklet_struct *t)
    {
    	if (!test_and_set_bit(TASKLET_STATE_SCHED, &t->state))
    		__tasklet_hi_schedule(t);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5

    kernel/softirq.c

    void __tasklet_hi_schedule(struct tasklet_struct *t)
    {
    	__tasklet_schedule_common(t, &tasklet_hi_vec,
    				  HI_SOFTIRQ);
    }
    EXPORT_SYMBOL(__tasklet_hi_schedule);
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    调度tasklet进入high优先级调度队列是通过HI_SOFTIRQ软中断触发

    Based on softirq

    几个软中断定义在include/linux/interrupt.h

    /* PLEASE, avoid to allocate new softirqs, if you need not _really_ high
       frequency threaded job scheduling. For almost all the purposes
       tasklets are more than enough. F.e. all serial device BHs et
       al. should be converted to tasklets, not to softirqs.
     */
    
    enum
    {
    	HI_SOFTIRQ=0,
    	TIMER_SOFTIRQ,
    	NET_TX_SOFTIRQ,
    	NET_RX_SOFTIRQ,
    	BLOCK_SOFTIRQ,
    	IRQ_POLL_SOFTIRQ,
    	TASKLET_SOFTIRQ,
    	SCHED_SOFTIRQ,
    	HRTIMER_SOFTIRQ,
    	RCU_SOFTIRQ,    /* Preferable RCU should always be the last softirq */
    
    	NR_SOFTIRQS
    };
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    Kill tasklet
    void tasklet_kill(struct tasklet_struct *t)
    {
    	if (in_interrupt())
    		pr_notice("Attempt to kill tasklet from interrupt\n");
    
    	while (test_and_set_bit(TASKLET_STATE_SCHED, &t->state))
    		wait_var_event(&t->state, !test_bit(TASKLET_STATE_SCHED, &t->state));
    
    	tasklet_unlock_wait(t);
    	tasklet_clear_sched(t);
    }
    EXPORT_SYMBOL(tasklet_kill);
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    示例

    static method

    通过DECLARE_TASKLET()宏构建tasklet

    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include                 //kmalloc()
    #include              //copy_to/from_user()
    #include 
    #include 
    #include 
    #include 
    #include  
     
    #define IRQ_NO 11
     
    void tasklet_fn(struct tasklet_struct *); 
    /* Init the Tasklet by Static Method */
    DECLARE_TASKLET(test_tasklet, tasklet_fn);
     
     
    /*Tasklet Function*/
    void tasklet_fn(struct tasklet_struct *t)
    {
            printk(KERN_INFO "Executing Tasklet Function : tasklet count = %d\n", atomic_read(&t->count));
    }
     
     
    //Interrupt handler for IRQ 11. 
    static irqreturn_t irq_handler(int irq,void *dev_id) {
            printk(KERN_INFO "Shared IRQ: Interrupt Occurred");
            /*Scheduling Task to Tasklet*/
            tasklet_schedule(&test_tasklet); 
            
            return IRQ_HANDLED;
    }
     
    volatile int tasklet_value = 0;
     
    dev_t dev = 0;
    static struct class *dev_class;
    static struct cdev tasklet_cdev;
    struct kobject *kobj_ref;
     
    static int __init tasklet_driver_init(void);
    static void __exit tasklet_driver_exit(void);
     
    /*************** Driver Functions **********************/
    static int tasklet_open(struct inode *inode, struct file *file);
    static int tasklet_release(struct inode *inode, struct file *file);
    static ssize_t tasklet_read(struct file *filp, 
                    char __user *buf, size_t len,loff_t * off);
    static ssize_t tasklet_write(struct file *filp, 
                    const char *buf, size_t len, loff_t * off);
     
    /*************** Sysfs Functions **********************/
    static ssize_t sysfs_show(struct kobject *kobj, 
                    struct kobj_attribute *attr, char *buf);
    static ssize_t sysfs_store(struct kobject *kobj, 
                    struct kobj_attribute *attr,const char *buf, size_t count);
     
    struct kobj_attribute tasklet_attr = __ATTR(tasklet_value, 0660, sysfs_show, sysfs_store);
     
    /*
    ** File operation sturcture
    */
    static struct file_operations fops =
    {
            .owner          = THIS_MODULE,
            .read           = tasklet_read,
            .write          = tasklet_write,
            .open           = tasklet_open,
            .release        = tasklet_release,
    };
     
    /*
    ** This function will be called when we read the sysfs file
    */  
    static ssize_t sysfs_show(struct kobject *kobj, 
                    struct kobj_attribute *attr, char *buf)
    {
            printk(KERN_INFO "Sysfs - Read!!!\n");
            return sprintf(buf, "%d", tasklet_value);
    }
    /*
    ** This function will be called when we write the sysfsfs file
    */  
    static ssize_t sysfs_store(struct kobject *kobj, 
                    struct kobj_attribute *attr,const char *buf, size_t count)
    {
            printk(KERN_INFO "Sysfs - Write!!!\n");
            sscanf(buf,"%d",&tasklet_value);
            return count;
    }
    /*
    ** This function will be called when we open the Device file
    */  
    static int tasklet_open(struct inode *inode, struct file *file)
    {
            printk(KERN_INFO "Device File Opened...!!!\n");
            return 0;
    }
    /*
    ** This function will be called when we close the Device file
    */   
    static int tasklet_release(struct inode *inode, struct file *file)
    {
            printk(KERN_INFO "Device File Closed...!!!\n");
            return 0;
    }
    /*
    ** This function will be called when we read the Device file
    */ 
    static ssize_t tasklet_read(struct file *filp, 
                    char __user *buf, size_t len, loff_t *off)
    {
            struct irq_desc *desc;
            printk(KERN_INFO "Read function\n");
            desc = irq_to_desc(11);
            if (!desc)
            {
                return -EINVAL;
            }
            __this_cpu_write(vector_irq[59], desc);
            asm("int $0x3B");  // Corresponding to irq 11
            return 0;
    }
    /*
    ** This function will be called when we write the Device file
    */
    static ssize_t tasklet_write(struct file *filp, 
                    const char __user *buf, size_t len, loff_t *off)
    {
            printk(KERN_INFO "Write Function\n");
            return len;
    }
     
    /*
    ** Module Init function
    */ 
    static int __init tasklet_driver_init(void)
    {
            /*Allocating Major number*/
            if((alloc_chrdev_region(&dev, 0, 1, "tasklet_Dev")) <0){
                    printk(KERN_INFO "Cannot allocate major number\n");
                    return -1;
            }
            printk(KERN_INFO "Major = %d Minor = %d \n",MAJOR(dev), MINOR(dev));
     
            /*Creating cdev structure*/
            cdev_init(&tasklet_cdev,&fops);
     
            /*Adding character device to the system*/
            if((cdev_add(&tasklet_cdev,dev,1)) < 0){
                printk(KERN_INFO "Cannot add the device to the system\n");
                goto r_class;
            }
     
            /*Creating struct class*/
            if((dev_class = class_create(THIS_MODULE,"tasklet_class")) == NULL){
                printk(KERN_INFO "Cannot create the struct class\n");
                goto r_class;
            }
     
            /*Creating device*/
            if((device_create(dev_class,NULL,dev,NULL,"tasklet_device")) == NULL){
                printk(KERN_INFO "Cannot create the Device 1\n");
                goto r_device;
            }
     
            /*Creating a directory in /sys/kernel/ */
            kobj_ref = kobject_create_and_add("tasklet_sysfs",kernel_kobj);
     
            /*Creating sysfs file for tasklet_value*/
            if(sysfs_create_file(kobj_ref,&tasklet_attr.attr)){
                    printk(KERN_INFO"Cannot create sysfs file......\n");
                    goto r_sysfs;
            }
            if (request_irq(IRQ_NO, irq_handler, IRQF_SHARED, "tasklet_device", (void *)(irq_handler))) {
                printk(KERN_INFO "my_device: cannot register IRQ ");
                        goto irq;
            }
     
            printk(KERN_INFO "Device Driver Insert...Done!!!\n");
            return 0;
     
    irq:
            free_irq(IRQ_NO,(void *)(irq_handler));
     
    r_sysfs:
            kobject_put(kobj_ref); 
            sysfs_remove_file(kernel_kobj, &tasklet_attr.attr);
     
    r_device:
            class_destroy(dev_class);
    r_class:
            unregister_chrdev_region(dev,1);
            cdev_del(&tasklet_cdev);   
            return -1;
    }
    /*
    ** Module exit function
    */  
    static void __exit tasklet_driver_exit(void)
    {
            /*Kill the Tasklet */ 
            tasklet_kill(&test_tasklet);
            free_irq(IRQ_NO,(void *)(irq_handler));
            kobject_put(kobj_ref); 
            sysfs_remove_file(kernel_kobj, &tasklet_attr.attr);
            device_destroy(dev_class,dev);
            class_destroy(dev_class);
            cdev_del(&tasklet_cdev);
            unregister_chrdev_region(dev, 1);
            printk(KERN_INFO "Device Driver Remove...Done!!!\n");
    }
     
    module_init(tasklet_driver_init);
    module_exit(tasklet_driver_exit);
     
    MODULE_LICENSE("GPL");
    MODULE_DESCRIPTION("A simple device driver - Tasklet Static");
    MODULE_VERSION("1.15");
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135
    • 136
    • 137
    • 138
    • 139
    • 140
    • 141
    • 142
    • 143
    • 144
    • 145
    • 146
    • 147
    • 148
    • 149
    • 150
    • 151
    • 152
    • 153
    • 154
    • 155
    • 156
    • 157
    • 158
    • 159
    • 160
    • 161
    • 162
    • 163
    • 164
    • 165
    • 166
    • 167
    • 168
    • 169
    • 170
    • 171
    • 172
    • 173
    • 174
    • 175
    • 176
    • 177
    • 178
    • 179
    • 180
    • 181
    • 182
    • 183
    • 184
    • 185
    • 186
    • 187
    • 188
    • 189
    • 190
    • 191
    • 192
    • 193
    • 194
    • 195
    • 196
    • 197
    • 198
    • 199
    • 200
    • 201
    • 202
    • 203
    • 204
    • 205
    • 206
    • 207
    • 208
    • 209
    • 210
    • 211
    • 212
    • 213
    • 214
    • 215
    • 216
    • 217
    • 218
    • 219
    • 220
    • 221
    • 222
    • 223
    • 224
    dynamic method

    通过tasklet_init()函数构建tasklet

    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include                 //kmalloc()
    #include              //copy_to/from_user()
    #include 
    #include 
    #include 
    #include 
    #include 
     
     
    #define IRQ_NO 11
     
    void tasklet_fn(unsigned long); 
    /* Tasklet by Dynamic Method */
    struct tasklet_struct *dynamic_tasklet = NULL;
     
     
    /*Tasklet Function*/
    void tasklet_fn(unsigned long arg)
    {
            printk(KERN_INFO "Executing Tasklet Function : arg = %ld\n", arg);
    }
     
     
    //Interrupt handler for IRQ 11. 
    static irqreturn_t irq_handler(int irq,void *dev_id) {
            printk(KERN_INFO "Shared IRQ: Interrupt Occurred");
            /*Scheduling Task to Tasklet*/
            tasklet_schedule(dynamic_tasklet); 
            
            return IRQ_HANDLED;
    }
     
     
    volatile int tasklet_value = 0;
     
     
    dev_t dev = 0;
    static struct class *dev_class;
    static struct cdev tasklet_cdev;
    struct kobject *kobj_ref;
     
    static int __init tasklet_driver_init(void);
    static void __exit tasklet_driver_exit(void);
     
    /*************** Driver Functions **********************/
    static int tasklet_open(struct inode *inode, struct file *file);
    static int tasklet_release(struct inode *inode, struct file *file);
    static ssize_t tasklet_read(struct file *filp, 
                    char __user *buf, size_t len,loff_t * off);
    static ssize_t tasklet_write(struct file *filp, 
                    const char *buf, size_t len, loff_t * off);
     
    /*************** Sysfs Functions **********************/
    static ssize_t sysfs_show(struct kobject *kobj, 
                    struct kobj_attribute *attr, char *buf);
    static ssize_t sysfs_store(struct kobject *kobj, 
                    struct kobj_attribute *attr,const char *buf, size_t count);
     
    struct kobj_attribute tasklet_attr = __ATTR(tasklet_value, 0660, sysfs_show, sysfs_store);
    /*
    ** File operation sturcture
    */
    static struct file_operations fops =
    {
            .owner          = THIS_MODULE,
            .read           = tasklet_read,
            .write          = tasklet_write,
            .open           = tasklet_open,
            .release        = tasklet_release,
    };
    /*
    ** This function will be called when we read the sysfs file
    */
    static ssize_t sysfs_show(struct kobject *kobj, 
                    struct kobj_attribute *attr, char *buf)
    {
            printk(KERN_INFO "Sysfs - Read!!!\n");
            return sprintf(buf, "%d", tasklet_value);
    }
    /*
    ** This function will be called when we write the sysfsfs file
    */   
    static ssize_t sysfs_store(struct kobject *kobj, 
                    struct kobj_attribute *attr,const char *buf, size_t count)
    {
            printk(KERN_INFO "Sysfs - Write!!!\n");
            sscanf(buf,"%d",&tasklet_value);
            return count;
    }
    /*
    ** This function will be called when we open the Device file
    */   
    static int tasklet_open(struct inode *inode, struct file *file)
    {
            printk(KERN_INFO "Device File Opened...!!!\n");
            return 0;
    }
     
    /*
    ** This function will be called when we close the Device file
    */   
    static int tasklet_release(struct inode *inode, struct file *file)
    {
            printk(KERN_INFO "Device File Closed...!!!\n");
            return 0;
    }
    /*
    ** This function will be called when we read the Device file
    */  
    static ssize_t tasklet_read(struct file *filp, 
                    char __user *buf, size_t len, loff_t *off)
    {
            struct irq_desc *desc;
            printk(KERN_INFO "Read function\n");
            desc = irq_to_desc(11);
            if (!desc)
            {
                return -EINVAL;
            }
            __this_cpu_write(vector_irq[59], desc);
            asm("int $0x3B");  // Corresponding to irq 11
            return 0;
    }
    /*
    ** This function will be called when we write the Device file
    */
    static ssize_t tasklet_write(struct file *filp, 
                    const char __user *buf, size_t len, loff_t *off)
    {
            printk(KERN_INFO "Write Function\n");
            return len;
    }
     
    /*
    ** Module Init function
    */ 
    static int __init tasklet_driver_init(void)
    {
            /*Allocating Major number*/
            if((alloc_chrdev_region(&dev, 0, 1, "tasklet_Dev_dynamic")) <0){
                    printk(KERN_INFO "Cannot allocate major number\n");
                    return -1;
            }
            printk(KERN_INFO "Major = %d Minor = %d \n",MAJOR(dev), MINOR(dev));
     
            /*Creating cdev structure*/
            cdev_init(&tasklet_cdev,&fops);
     
            /*Adding character device to the system*/
            if((cdev_add(&tasklet_cdev,dev,1)) < 0){
                printk(KERN_INFO "Cannot add the device to the system\n");
                goto r_class;
            }
     
            /*Creating struct class*/
            if((dev_class = class_create(THIS_MODULE,"tasklet_class_dynamic")) == NULL){
                printk(KERN_INFO "Cannot create the struct class\n");
                goto r_class;
            }
     
            /*Creating device*/
            if((device_create(dev_class,NULL,dev,NULL,"tasklet_device_dynamic")) == NULL){
                printk(KERN_INFO "Cannot create the Device 1\n");
                goto r_device;
            }
     
            /*Creating a directory in /sys/kernel/ */
            kobj_ref = kobject_create_and_add("tasklet_sysfs_dynamic",kernel_kobj);
     
            /*Creating sysfs file for tasklet_value*/
            if(sysfs_create_file(kobj_ref,&tasklet_attr.attr)){
                    printk(KERN_INFO"Cannot create sysfs file......\n");
                    goto r_sysfs;
            }
            if (request_irq(IRQ_NO, irq_handler, IRQF_SHARED, "tasklet_device_dynamic", (void *)(irq_handler))) {
                printk(KERN_INFO "tasklet_device_dynamic: cannot register IRQ ");
                goto irq;
            }
            /* Init the tasklet bt Dynamic Method */
            dynamic_tasklet  = kmalloc(sizeof(struct tasklet_struct),GFP_KERNEL);
            if(dynamic_tasklet == NULL) {
                printk(KERN_INFO "tasklet_device_dynamic: cannot allocate Memory");
                goto irq;
            }
            tasklet_init(dynamic_tasklet,tasklet_fn,0);
     
            printk(KERN_INFO "Device Driver Insert...Done!!!\n");
            return 0;
     
    irq:
            free_irq(IRQ_NO,(void *)(irq_handler));
     
    r_sysfs:
            kobject_put(kobj_ref); 
            sysfs_remove_file(kernel_kobj, &tasklet_attr.attr);
     
    r_device:
            class_destroy(dev_class);
    r_class:
            unregister_chrdev_region(dev,1);
            cdev_del(&tasklet_cdev);        
            return -1;
    }
    /*
    ** Module exit function
    */ 
    static void __exit tasklet_driver_exit(void)
    {
            /* Kill the Tasklet */ 
            tasklet_kill(dynamic_tasklet);
            if(dynamic_tasklet != NULL)
            {
              kfree(dynamic_tasklet);
            }
            free_irq(IRQ_NO,(void *)(irq_handler));
            kobject_put(kobj_ref); 
            sysfs_remove_file(kernel_kobj, &tasklet_attr.attr);
            device_destroy(dev_class,dev);
            class_destroy(dev_class);
            cdev_del(&tasklet_cdev);
            unregister_chrdev_region(dev, 1);
            printk(KERN_INFO "Device Driver Remove...Done!!!\n");
    }
     
    module_init(tasklet_driver_init);
    module_exit(tasklet_driver_exit);
     
    MODULE_LICENSE("GPL");
    MODULE_DESCRIPTION("A simple device driver - Tasklet Dynamic");
    MODULE_VERSION("1.16");
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135
    • 136
    • 137
    • 138
    • 139
    • 140
    • 141
    • 142
    • 143
    • 144
    • 145
    • 146
    • 147
    • 148
    • 149
    • 150
    • 151
    • 152
    • 153
    • 154
    • 155
    • 156
    • 157
    • 158
    • 159
    • 160
    • 161
    • 162
    • 163
    • 164
    • 165
    • 166
    • 167
    • 168
    • 169
    • 170
    • 171
    • 172
    • 173
    • 174
    • 175
    • 176
    • 177
    • 178
    • 179
    • 180
    • 181
    • 182
    • 183
    • 184
    • 185
    • 186
    • 187
    • 188
    • 189
    • 190
    • 191
    • 192
    • 193
    • 194
    • 195
    • 196
    • 197
    • 198
    • 199
    • 200
    • 201
    • 202
    • 203
    • 204
    • 205
    • 206
    • 207
    • 208
    • 209
    • 210
    • 211
    • 212
    • 213
    • 214
    • 215
    • 216
    • 217
    • 218
    • 219
    • 220
    • 221
    • 222
    • 223
    • 224
    • 225
    • 226
    • 227
    • 228
    • 229
    • 230
    • 231
    • 232
    • 233
    • 234
    • 235
    • 236
    • 237

    reference

    Linux Device Driver Tutorial – ch20~21

  • 相关阅读:
    9-FreeRTOS之静态内存分配与动态内存分配
    DC/DC开关电源学习笔记(十二)Boost升压电路仿真及工程应用案例
    推荐系统-排序层-2018:ESMM【多任务学习模型】【多任务学习(multi-task learning,简写MTL)】【阿里】
    【Linux篇】第十七篇——信号量
    Android开发基础:Activity的生命周期 Activity中的数据保持
    【博客547】keepalived实现vip的原理剖析
    入门JavaWeb之 Response 下载文件
    CPU vs GPU:谁更适合进行图像处理?
    C++初学者指南第一步---2. Hello world
    TcpServerChannel 类服务
  • 原文地址:https://blog.csdn.net/qq_23662505/article/details/126889057