0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

使用Linux自旋锁实现互斥点灯

CHANBAEK 来源:嵌入式攻城狮 作者: 安迪西 2023-04-13 15:09 次阅读

1. 自旋锁介绍

自旋锁最多只能被一个可执行线程持有。 如果一个线程试图获得一个已经被持有的自旋锁,那么该线程将循环等待,然后不断的判断锁是否能够被成功获取,直到获取到锁才会退出循环; 如果锁未被持有,请求锁的执行线程就可以立即得到它,继续执行

等待自旋锁的线程一直处于自旋状态,会浪费处理器时间,降低系统性能,因此自旋锁的持有时间不能太长,适用于短时期的轻量级加锁

内核使用结构体spinlock_t表示自旋锁,结构体定义如下所示:

typedef struct spinlock {
    union {
        struct raw_spinlock rlock;
#ifdef CONFIG_DEBUG_LOCK_ALLOC
#define LOCK_PADSIZE (offsetof(struct raw_spinlock, dep_map))
        struct {
            u8 __padding[LOCK_PADSIZE];
            struct lockdep_map dep_map;
        };
#endif
    };
} spinlock_t;

自旋锁操作相关API函数如下图示:

图片

2. 自旋锁实例

本实例中使用自旋锁操作来实现对LED设备的互斥访问,即一次只允许一个应用程序使用LED灯,代码是在pinctrl与gpio子系统下的字符设备驱动框架一文的基础上完成的

在本例程中,定义一个变量dev_stats表示设备的使用情况,该变量为0时表示设备没有被使用,大于0时表示设备被使用。 在驱动的open函数中需先判断该变量是否为0,若为0的话就使用设备,并且将dev_stats加1,表示设备被使用了。 使用完后在release函数中需将dev_stats减1,表示设备没有被使用了。 因此真正实现设备互斥访问的是变量dev_stats,我们使用自旋锁对dev_stats来做保护

2.1 修改设备树文件

设备树文件修改与pinctrl与gpio子系统下的字符设备驱动框架文中的修改方法一样,不需要做任何修改

2.2 编写驱动程序

拷贝pinctrl与gpio子系统下的字符设备驱动框架文中的gpioled.c驱动文件,并重命名为spinlock.c,对部分代码进行修改,其余保持不变

在设备结构体中,添加自旋锁以及设备状态变量dev_stats

struct gpioled_dev{
    dev_t devid;               //设备号
    struct cdev cdev;          //cdev字符设备
    struct class *class;       //类
    struct device *device;     //设备
    int major;                 //主设备号
    int minor;                 //次设备号
    struct device_node *nd;    //设备节点
    int led_gpio;              //所使用的gpio编号
    int dev_stats;             //设备状态,为0表示设备为使用
    spinlock_t lock;           //自旋锁
};

struct gpioled_dev gpioled;    //定义led设备

打开设备时,判断dev_stats的值来检查LED有没有被占用

static int led_open(struct inode *inode, struct file *filp){
    unsigned long flags;
    filp->private_data = &gpioled;             //设置私有数据

    spin_lock_irqsave(&gpioled.lock, flags);           //上锁
    if(gpioled.dev_stats)){
        spin_unlock_irqrestore(&gpioled.lock, flags);  //解锁
        return -EBUSY;                     //LED被使用,返回忙
    }
    gpioled.dev_stats++;                   //标记设备以打开
    spin_unlock_irqrestore(&gpioled.lock, flags);      //解锁
 
    return 0;
}

关闭设备时,将dev_stats减1,表示设备没有被使用了

static int led_release(struct inode *inode, struct file *filp){
    unsigned long flags;
    struct gpioled_dev *dev = filp->private_data;
 
    spin_lock_irqsave(&dev->lock, flags);          //上锁
    if(dev->dev_stats)){
        dev->dev_stats--;
    }
    spin_unlock_irqrestore(&dev->lock, flags);     //解锁

    return 0;
}

驱动入口函数中,对自旋锁进行初始化

static int __init led_init(void){
    int ret = 0;
    /* 初始化自旋锁 */
    spin_lock_init(&gpioled.lock);  
    /* 设置 LED 所使用的 GPIO */
    /* 1、获取设备节点:gpioled */
    gpioled.nd = of_find_node_by_path("/gpioled");
    ......
    ......
}

2.3 编写测试程序

拷贝pinctrl与gpio子系统下的字符设备驱动框架文中的gpioledApp.c测试程序,并重命名为spinlockApp.c,添加模拟占用LED的代码,使测试程序在获取LED驱动使用权后会持续一段时间,添加如下代码

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

2.4 编译测试

编译驱动程序:当前目录下创建Makefile文件,并make编译

KERNELDIR := /home/andyxi/linux/kernel/linux-imx-rel_imx_4.1.15_2.1.0_ga_andyxi
CURRENT_PATH := $(shell pwd)
obj-m := spinlock.o

build: kernel_modules

kernel_modules:
$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) modules
clean:
$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) clean

编译测试程序:无需内核参与,直接编译即可

arm-linux-gnueabihf-gcc spinlockApp.c -o spinlockApp

运行测试:启动开发板后,加载驱动模块,操作LED灯后,相应时间内再次操作LED会提示失败,说明互斥点灯成功

depmod                   #第一次加载驱动的时候需要运行此命令
modprobe spinlock.ko      #加载驱动
# 打开LED后,每隔5秒会输出一行App running times
./spinlockApp /dev/gpioled 1&   # & 表示在后台运行APP

图片

# 在LED被占用期间,再次操作LED,会输出打开驱动失败
./spinlockApp /dev/gpioled 0

图片

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Linux
    +关注

    关注

    87

    文章

    11322

    浏览量

    209869
  • 子系统
    +关注

    关注

    0

    文章

    109

    浏览量

    12417
  • 函数
    +关注

    关注

    3

    文章

    4338

    浏览量

    62769
  • 线程
    +关注

    关注

    0

    文章

    505

    浏览量

    19711
  • 自旋锁
    +关注

    关注

    0

    文章

    11

    浏览量

    1607
收藏 人收藏

    评论

    相关推荐

    深度解析自旋自旋实现方案

    入场券自旋和MCS自旋都属于排队自旋(queued spinlock),进程按照申请
    发表于 09-19 11:39 4449次阅读
    深度解析<b class='flag-5'>自旋</b><b class='flag-5'>锁</b>及<b class='flag-5'>自旋</b><b class='flag-5'>锁</b>的<b class='flag-5'>实现</b>方案

    Linux高级编程---互斥

    Linux系统里,有很多的应用,包括互斥,文件,读写等等,信号量其实也应该是
    发表于 01-13 10:07

    Linux驱动开发笔记-自旋和信号量

    :如果在写代码时,有以上的竞态发生,一定要注意进行互斥访问7.解决竞态的方法:中断屏蔽原子操作自旋信号量如何使用以上4个机制呢?1.中断屏蔽解决哪些情况的竞态:进程和进程的抢占中断和进程中断和中断
    发表于 08-30 18:08

    信号量、互斥自旋

    信号量、互斥自旋http://bbs.edu118.com/forum.php?mod=viewthread&tid=488&fromuid=231(出处: 信盈达IT技术社
    发表于 08-29 09:48

    怎么在atmega128中实现自旋

    什么是自旋?有哪些缺陷?怎么在atmega128中实现自旋
    发表于 01-24 06:54

    Linux 自旋spinlock

    背景 由于在多处理器环境中某些资源的有限性,有时需要互斥访问(mutual exclusion),这时候就需要引入的概念,只有获取了的任务才能够对资源进行访问,由于多线程的核心是CPU的时间分片
    的头像 发表于 09-11 14:36 2099次阅读

    Linux中的伤害/等待互斥介绍

    序言:近期读Linux 5.15的发布说明,该版本合并了实时机制,当开启配置宏CONFIG_PREEMPT_RT的时候,这些被基于实时互斥
    的头像 发表于 11-06 17:27 2686次阅读

    Linux下线程间通讯--互斥

    互斥是一种简单的加锁的方法来控制对共享资源的存取,当多个线程访问公共资源时,为了保证同一时刻只有一个线程独占资源,就可以通过互斥加以限制,在一个时刻只能有一个线程掌握某个
    的头像 发表于 08-24 15:53 1983次阅读
    <b class='flag-5'>Linux</b>下线程间通讯--<b class='flag-5'>互斥</b><b class='flag-5'>锁</b>

    使用Linux互斥实现互斥点灯

    互斥访问是指一次只有一个线程可以访问共享资源,不能递归申请互斥体。使用互斥体时要注意如下几点。
    的头像 发表于 04-13 15:13 877次阅读
    使用<b class='flag-5'>Linux</b><b class='flag-5'>互斥</b>体<b class='flag-5'>实现</b><b class='flag-5'>互斥</b><b class='flag-5'>点灯</b>

    Linux互斥的作用 互斥是什么

    1、互斥 互斥(mutex),在访问共享资源之前对互斥进行上锁,在访问完成后释放
    的头像 发表于 07-21 11:13 961次阅读

    自旋互斥的区别有哪些

    自旋 自旋互斥很相似,在访问共享资源之前对自旋
    的头像 发表于 07-21 11:19 9520次阅读

    如何用C++11实现自旋

    下面我会分析一下自旋,并代码实现自旋互斥的性
    的头像 发表于 11-11 16:48 1469次阅读
    如何用C++11<b class='flag-5'>实现</b><b class='flag-5'>自旋</b><b class='flag-5'>锁</b>

    互斥自旋的区别 自旋临界区可以被中断吗?

    互斥自旋的区别 自旋临界区可以被中断吗? 互斥
    的头像 发表于 11-22 17:41 862次阅读

    自旋互斥的使用场景是什么

    自旋互斥是两种常见的同步机制,它们在多线程编程中被广泛使用。在本文中,我们将介绍自旋
    的头像 发表于 07-10 10:05 1048次阅读

    互斥自旋实现原理

    互斥自旋是操作系统中常用的同步机制,用于控制对共享资源的访问,以避免多个线程或进程同时访问同一资源,从而引发数据不一致或竞争条件等问题。 互斥
    的头像 发表于 07-10 10:07 529次阅读