Linux驱动开发七.并发与竞争——2.实际操作

我们在前面讲了处理竞争和并发问题的四种机制,下面可以通过一些驱动来检验一下。

原子操作

原子操作用了最基础的一个虚拟的设备来演示,在设备模块被加载后生成了设备节点,我们使用APP程序打开设备节点后是有个线程访问了该设备里的数据,当另外一个APP重新要打开这个数据时就无法正常访问了。

  1 /**
  2  * @file atomic.c
  3  * @author your name (you@domain.com)
  4  * @brief 原子变量测试
  5  * @version 0.1
  6  * @date 2022-07-07
  7  * 
  8  * @copyright Copyright (c) 2022
  9  * 
 10  */
 11 #include <linux/module.h>
 12 #include <linux/kernel.h>
 13 #include <linux/init.h>
 14 #include <linux/fs.h>
 15 #include <linux/uaccess.h>
 16 #include <linux/io.h>
 17 #include <linux/types.h>
 18 #include <linux/cdev.h>
 19 #include <linux/device.h>
 20 #include <linux/of.h>
 21 #include <linux/of_address.h>
 22 #include <linux/of_irq.h>
 23 #include <linux/gpio.h>
 24 #include <linux/of_gpio.h>
 25 #include <linux/atomic.h>
 26 
 27 #define DEVICE_CNT      1
 28 #define DEVICE_NAME    "DEV"
 29 
 30 struct test_dev
 31 {
 32     dev_t dev_id;
 33     int major;
 34     int minor;
 35     struct class *class;
 36     struct device *device;
 37     struct cdev cdev;
 38     struct device_node *dev_nd;
 39     atomic_t atomic_data;              //原子变量
 40 };
 41 
 42 struct test_dev dev;
 43 
 44 static ssize_t dev_open(struct inode *inode, struct file *filp)
 45 {
 46     filp->private_data = &dev;                  //设置私有数据
 47     //首次打开文件时,流程为else,原子变量值减1为0,再次打开后值为0走if流程,返回-EBUSY状态
 48     if(atomic_read(&dev.atomic_data) <= 0){     //获取原子变量值
 49         return -EBUSY;
 50     }
 51     else{
 52         atomic_dec(&dev.atomic_data);
 53     }
 54     return 0;
 55 }
 56 
 57 static ssize_t dev_write(struct file *filp,const char __user *buf,
 58                             size_t count,loff_t *ppos)
 59 {
 60     int ret = 0;
 61     return ret;
 62 }
 63 
 64 static int dev_release(struct inode *inode,struct file *filp)
 65 {
 66     //文件关闭,原子变量自增(打开文件时减一为0,关闭后变成1,可以再次打开文件)
 67     struct test_dev *dev = filp->private_data;
 68     atomic_inc(&dev->atomic_data);
 69     return 0;
 70 }
 71 
 72 /**
 73  * @brief 文件操作集合
 74  * 
 75  */
 76 static const struct file_operations gpiofops = {
 77     .owner = THIS_MODULE,
 78     .open =  dev_open,
 79     .release = dev_release,
 80 };
 81 
 82 static int __init dev_init(void){
 83 
 84     int ret = 0; 
 85     //初始化原子变量,赋值为1
 86     atomic_set(&dev.atomic_data,1);
 87 
 88     //申请设备号
 89     dev.major = 0;
 90     if(dev.major){
 91         //手动指定设备号,使用指定的设备号
 92         dev.dev_id = MKDEV(dev.major,0);
 93         ret = register_chrdev_region(dev.dev_id,DEVICE_CNT,DEVICE_NAME);
 94     }
 95     else{
 96         //设备号未指定,申请设备号
 97         ret = alloc_chrdev_region(&dev.dev_id,0,DEVICE_CNT,DEVICE_NAME);
 98         dev.major = MAJOR(dev.dev_id);
 99         dev.minor = MINOR(dev.dev_id);
100     }
101     printk("dev id geted!\r\n");
102 
103     if(ret<0){
104         //设备号申请异常,跳转至异常处理
105         goto faile_devid;
106     }
107 
108     //字符设备cdev初始化
109     dev.cdev.owner = THIS_MODULE;
110 
111     cdev_init(&dev.cdev,&gpiofops);                 //文件操作集合映射
112 
113     ret = cdev_add(&dev.cdev,dev.dev_id,DEVICE_CNT);
114     if(ret<0){
115         //cdev初始化异常,跳转至异常处理
116         goto fail_cdev;
117     }
118 
119     printk("chr dev inited!\r\n");
120 
121     //自动创建设备节点
122     dev.class = class_create(THIS_MODULE,DEVICE_NAME);
123     if(IS_ERR(dev.class)){
124         //class创建异常处理
125         printk("class err!\r\n");
126         ret = PTR_ERR(dev.class);
127         goto fail_class;
128     }
129     printk("dev class created\r\n");
130     dev.device = device_create(dev.class,NULL,dev.dev_id,NULL,DEVICE_NAME);
131     if(IS_ERR(dev.device)){
132         //设备创建异常处理
133         printk("device err!\r\n");
134         ret = PTR_ERR(dev.device);
135         goto fail_device;
136     }
137     printk("device created!\r\n");
138 
139     return 0;
140 
141 fail_device:
142     //device创建失败,意味着class创建成功,应该将class销毁
143     printk("device create err,class destroyed\r\n");
144     class_destroy(dev.class);
145 fail_class:
146     //类创建失败,意味着设备应该已经创建成功,此刻应将其释放掉
147     printk("class create err,cdev del\r\n");
148     cdev_del(&dev.cdev);
149 fail_cdev:
150     //cdev初始化异常,意味着设备号已经申请完成,应将其释放
151     printk("cdev init err,chrdev register\r\n");
152     unregister_chrdev_region(dev.dev_id,DEVICE_CNT);
153 faile_devid:
154     //设备号申请异常,由于是第一步操作,不需要进行其他处理
155     printk("dev id err\r\n");
156     return ret;
157 }
158 
159 static void __exit dev_exit(void)
160 {
161     cdev_del(&dev.cdev);
162     unregister_chrdev_region(dev.dev_id,DEVICE_CNT);
163 
164     device_destroy(dev.class,dev.dev_id);
165     class_destroy(dev.class);
166 }
167 
168 module_init(dev_init);
169 module_exit(dev_exit);
170 
171 MODULE_LICENSE("GPL");
172 MODULE_AUTHOR("ZeqiZ");

整个程序比较简单,就是在驱动框架的基础上进行了写小变动,主要是文件操作的几个函数做了相应的修改。

第39行,在设备结构体中,我们定义了一个原子变量atomic_data,注意这个变量为整形变量

第86行,驱动模块在初始化过程中对原子变量进行初始化,并赋值为1

第44-55行,设备结构体dev被作为文件的私有数据,在文件首次被打开时,由于atomic_data初始值为1,文件能正常打开,文件打开后atomic_data进行自减操作变成0。此刻如果有另外一个线程进行了文件打开操作,atomic_data值为0,函数返回-EBUSY异常

第64到69行,设备文件被关闭,atomic_data进行自增操作,值重新为1,文件可以再次被打开。

这个APP需要修改一下

 1 /**
 2  * @file atomicAPP.c
 3  * @author your name (you@domain.com)
 4  * @brief 原子数据测试APP
 5  * @version 0.1
 6  * @date 2022-07-07
 7  * 
 8  * @copyright Copyright (c) 2022
 9  * 
10  */
11 #include <sys/types.h>
12 #include <sys/stat.h>
13 #include <fcntl.h>
14 #include <unistd.h>
15 #include <stdio.h>
16 #include <stdlib.h>
17 
18 /**
19  * @brief 
20  * 
21  * @param argc                      //参数个数 
22  * @param argv                      //参数
23  * @return int 
24  */
25 int main(int argc,char *argv[])
26 {
27     char *filename;                 //文件名
28     filename = argv[1];             //文件名为命令行后第二个参数(索引值为1)
29 
30     int ret = 0;                    //初始化操作返回值
31     int f = 0;                      //初始化文件句柄
32     int cnt=0;
33 
34     if(argc != 2){                  //输入参数个数不为3,提示输入格式错误
35         printf("input format error!\r\n");
36     }
37 
38     f = open(filename, O_RDWR);     //打开文件
39     if(f < 0){
40     printf("file open error\r\n");
41     return -1;
42     }
43 
44     while(1){
45         printf("file opened\r\n");
46         sleep(5);
47         cnt++;
48         printf("App running times:%d\r\n",cnt);
49         if(cnt>=3){
50             break;
51         }        
52     }
53 
54     printf("App running finished\r\n");
55 
56     close(f);                       //关闭文件
57     printf("file closed!\r\n");
58     return 0;
59 }

在APP里主要处理一个文件打开的操作,由于我们没发模拟线程访问数据的过程,只能通过开启文件然后延时来模拟线程持有数据的过程。然后用了个for循环3次显示程序正在运行。最后程序运行的效果如下图

我在运行程序时加了个&符号,意思是后台运行程序,运行期间再在终端重新使用命令打开文件,就会有错误提示。如果在运行程序期间使用ps命令打印进程,就可以看到我们运行的程序

 

注意上面那个图最下面还显示出来程序正在运行(APP里打印文件打开的语句应该放在for循环外面,这里实在不想改了。。。)

程序优化

这个上面的过程我们使用了下面两组原子变量的操作

atomic_dec(&dev.atomic_data);
atomic_inc(&dev.atomic_data);

另外还结合了if语句进行判定加上else来选择流程,但是Linux内核给我们提供了另一个API,在自增/减的时候直接判定

static ssize_t dev_open(struct inode *inode, struct file *filp)
{
    filp->private_data = &dev;                  //设置私有数据
    //首次打开文件时,流程为else,原子变量值减1为0,再次打开后值为0走if流程,返回-EBUSY状态  
#if 0
    if(atomic_read(&dev.atomic_data) <= 0){     //获取原子变量值
        return -EBUSY;
    }
    else{
        atomic_dec(&dev.atomic_data);
    }
#endif  
    if(!atomic_dec_and_test(&dev.atomic_data)){
        atomic_inc(&dev.atomic_data);
        return -EBUSY;
    }
    return 0;
}

注意函数操作顺序

atomic_dec_and_test(&dev.atomic_data)

先将变量减一,再判定,如果减一后结果为0则返回1,否则返回0。我们开始将变量初始化为1,进行减一后变成0,函数返回1,用来一个非运算,就不用执行if结构里的代码直接return,如果文件已经打开,变量值为0,自减后不为0,返回1,非运算后执行if结构体里内容,因为前面判定时减了1,所以要再加回去,返回异常值。这种结构是Linux内核中驱动开发的主要使用方法,在内核里搜索一下很多案例可以供我们参考。总之,这种整形的原子变量,在我们用作计数器的时候非常普遍。

自旋锁操作

自旋锁的演示和前面原子操作差不多,不放全部代码了,区别就是在定义设备结构体时要定义一个锁和一个状态字。

struct test_dev
{
    dev_t dev_id;
    int major;
    int minor;
    struct class *class;
    struct device *device;
    struct cdev cdev;
    struct device_node *dev_nd;

    int dev_status;     //设备状态,0时可被使用,1时不可使用
    spinlock_t lock;    //自旋锁
};

在设备初始化的时候要将锁和状态字初始化,

static int __init dev_init(void){

    int ret = 0; 
    //初始化自旋锁
    spin_lock_init(&dev.lock);
    dev.dev_status = 0;
    //.....后面省略

把状态初始化为0,表示0未上锁,1时上锁

锁的操作在open和release对应的函数中

static ssize_t dev_open(struct inode *inode, struct file *filp)
{
    filp->private_data = &dev; //设置私有数
    printk("open\r\n");

    spin_lock(&dev.lock);
    if(dev.dev_status){         //判定是否被占用
        spin_unlock(&dev.lock);
        return -EBUSY;
    }

    dev.dev_status++;          //自增,非0表示被占用
    spin_unlock(&dev.lock);

    return 0;
}


static int dev_release(struct inode *inode,struct file *filp)
{
    struct test_dev *dev = filp->private_data;
    spin_lock(&dev->lock);
    if(dev->dev_status){
        dev->dev_status--;     //未被占用,自减至0,表示未被占用
    }
    spin_unlock(&dev->lock);
    return 0;
}

注意看一下,我们在前面讲过,自旋锁是一种轻量锁,所以自旋锁锁定的是状态字而不是数据,如果是锁数据,当数据量很大的情况下自旋锁操作就不方便了。这就是我们在前面讲到的,自旋锁主要用来锁临界区。其实上面这种效果用原子操作是比较合适的,这么用主要是为了演示自旋锁的用法

这里应用演示和前面的效果一样,我就不在截图了(主要是没想好怎么模拟两条线程访问的方法)

自旋锁中断处理

在上一章里讲到过,因为系统中的中断是不好控制的,如果使用自旋锁最好将本地中断禁止掉,所以上面open和release两个函数最好使用中断禁止的自旋锁请求

static ssize_t dev_open(struct inode *inode, struct file *filp)
{
    unsigned long irqflag;
    filp->private_data = &dev; //设置私有数
    printk("open\r\n");
    spin_lock_irqsave(&dev.lock,irqflag);
    if(dev.dev_status){
        spin_unlock_irqrestore(&dev.lock);
        return -EBUSY;}
        //if条件为真,不能使用
    dev.dev_status++;
    spin_unlock_irqrestore(&dev.lock,irqflag);
    return 0;
}

static int dev_release(struct inode *inode,struct file *filp)
{
    struct test_dev *dev = filp->private_data;
    unsigned long irqflag;                      //中断标志
    spin_lock_irqsave(&dev->lock,irqflag);
    if(dev->dev_status){
        dev->dev_status--;
    }
    spin_unlock_irqrestore(&dev->lock,irqflag);
    return 0;
}

整个流程差不多,只用声明一个变量来描述中断标志就可以了。

信号量

信号量的使用要简单的多,直接看代码

struct test_dev
{
    dev_t dev_id;
    int major;
    int minor;
    struct class *class;
    struct device *device;
    struct cdev cdev;
    struct device_node *dev_nd;

    struct semaphore sem;        //信号量
};

struct test_dev dev;

static ssize_t dev_open(struct inode *inode, struct file *filp)
{
    filp->private_data = &dev; //设置私有数
    printk("open\r\n");
    down(&dev.sem);             //获取信号量
    
    return 0;
}

static int dev_release(struct inode *inode,struct file *filp)
{
    struct test_dev *dev = filp->private_data;

    up(&dev->sem);
return 0; }

在初始化函数要对变量进行初始化

static int __init dev_init(void){

    int ret = 0; 
    //初始化信号量
    sema_init(&dev.sem,1);

将信号量初始化为1,在打开文件时会down操作将信号量自减1,如果自减后变成0线程就会休眠,直到有up操作将信号量恢复到非0状态(注意变量类型,如果在定义设备结构体时没有加*指针后面使用时要加取址符,加了*后面初始化和up,down是就不加取址符了!还有关键字是semaphore,还有另外一个关键字跟他很像:semphore,用自动补全时一定要注意,当初在这里卡了半天!)。是不是简单的多!把APP程序稍微修改一下

/**
 * @file semaAPP.c
 * @author your name (you@domain.com)
 * @brief 
 * @version 0.1
 * @date 2022-07-10
 * 
 * @copyright Copyright (c) 2022
 * 
 */
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>


/**
 * @brief 
 * 
 * @param argc                      //参数个数 
 * @param argv                      //参数
 * @return int 
 */
int main(int argc,char *argv[])
{
    char *filename;                 //文件名
    filename = argv[1];             //文件名为命令行后第二个参数(索引值为1)
    
    char *id;
    id = argv[2];

    int ret = 0;                    //初始化操作返回值
    int f = 0;                      //初始化文件句柄

    int cnt=0;

    if(argc != 3){                  //输入参数个数不为3,提示输入格式错误
        printf("input format error!\r\n");
    }

    f = open(filename, O_RDWR);     //打开文件
    printf("id=%s file open\r\n",id);
    if(f < 0){
    printf("file open error\r\n");
    return -1;
    }

    while(1){
        sleep(5);
        cnt++;
        printf("App id = %s running times:%d\r\n",id,cnt);
        if(cnt>=5){
            break;
        }
        
    }

    printf("App id=%s running finished\r\n" ,id);

    close(f);                       //关闭文件
    return 0;
}

运行程序时,第三个参数为程序id,运行一下看看效果

 

运行APP测试一下,当我们第一次运行程序,程序正常执行,当第2个程序调用的时候,由于信号量初始值为1,就会被挂起来,直到第一个程序运行完后执行力up命令恢复了信号量后第二个程序开始运行,注意打印的id号。如果初始化信号量时候将信号量的值设置为大于1时,那么可以同时运行的程序就不止1个了。

互斥体

互斥体的使用方法和信号量差不多,把部分代码放下面

struct test_dev
{
    dev_t dev_id;
    int major;
    int minor;
    struct class *class;
    struct device *device;
    struct cdev cdev;
    struct device_node *dev_nd;

    struct mutex mut;
};

struct test_dev dev;

static ssize_t dev_open(struct inode *inode, struct file *filp)
{
    filp->private_data = &dev; //设置私有数
    printk("open\r\n");
    mutex_lock(&dev.mut);
    return 0;
}

static int dev_release(struct inode *inode,struct file *filp)
{
    struct test_dev *dev = filp->private_data;

    mutex_unlock(&dev->mut);

    return 0;
}

还有初始化

static int __init dev_init(void){

    int ret = 0; 
    //初始化
    mutex_init(&dev.mut);

使用效果跟前面的信号量值为1时是一样的。

信号量和互斥体还有个用法是在休眠时允许信号打断,这个用法以后用到的概率会比较大,这里就先不讲了,后面肯定会用到。

posted @ 2022-07-10 11:38  银色的音色  阅读(130)  评论(0编辑  收藏  举报