Github: atomic_fetch_add_unless
Email: BuddyZhang1 buddy.zhang@aliyun.com
Architecture: ARMv7 Cortex A9-MP
目录
源码分析
atomic_fetch_add_unless() 函数用于当 atomic_t 变量值与某个值不相等是,执行加法操作。
参数 v 指向一个 atomic_t 变量;参数 a 指明需要增加的数据;参数 u 代表与原始值对比的值。
函数首先调用 smp_mb() 函数添加一个内存屏障,接着调用 prefetchw() 函数预读取 v->counter
的值到 cache,接着调用内嵌汇编。在汇编中,首先调用 ldrex 指令添加独占标志,并将
v->counter 的值从内存读取到 oldval 变量里,然后通过对比 oldval 的值是否和 参数 u 相等,
如果相等,则跳转到 2 处;如果不相等,那么就将 oldval 的值与参数 a 的值相加,并把
相加的结果存储到 newval 里,接着调用 strex 指令,如果此时独占标志还存在,那么
strex 指令就将 newval 的值写入 v->counter 对应的内存,并肩 tmp 的值设置为 0;反之
如果此时独占标志已经被清零,那么 strex 指令仅仅将 tmp 设置为 1,然后继续执行
teq 指令,如果此时 teq 检查到 tmp 不为零,那么跳转到 1 处重复执行之前的动作,知道
将 newval 写入到 v->counter 对应的内存。最后如果 oldval 不等于 u,那么代表
strex 指令已经执行过了,所以此时需要在执行一条 smp_mb() 确保数据都写入到内存。
最后返回 oldval 的值。
实践
驱动源码
驱动安装
驱动的安装很简单,首先将驱动放到 drivers/BiscuitOS/ 目录下,命名为 atomic.c,
然后修改 Kconfig 文件,添加内容参考如下:
接着修改 Makefile,请参考如下修改:
驱动配置
驱动配置请参考下面文章中关于驱动配置一节。在配置中,勾选如下选项,如下:
具体过程请参考:
Linux 5.0 开发环境搭建 – 驱动配置
驱动编译
驱动编译也请参考下面文章关于驱动编译一节:
Linux 5.0 开发环境搭建 – 驱动编译
驱动运行
驱动的运行,请参考下面文章中关于驱动运行一节:
Linux 5.0 开发环境搭建 – 驱动运行
启动内核,并打印如下信息:
驱动分析
在某些有条件的 atomic 加法中,这个函数是比较适合的。
附录
BiscuitOS Home
BiscuitOS Driver
BiscuitOS Kernel Build
Linux Kernel
Bootlin: Elixir Cross Referencer
搭建高效的 Linux 开发环境
赞赏一下吧 🙂