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

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

3天内不再提示

那些书本上都没有提到的C语言volatile用法

STM32嵌入式开发 来源:嵌入式ARM 作者:嵌入式ARM 2021-10-12 14:47 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

许多程序员都无法正确理解C语言关键字volatile,这并不奇怪。因为大多数C语言书籍通常都是一两句一带而过,本文将告诉你如何正确使用它。

在C/C++嵌入式代码中,你是否经历过以下情况:

代码执行正常–直到你打开了编译器优化

代码执行正常–直到打开了中断

古怪的硬件驱动

RTOS的任务独立运行正常–直到生成了其他任务

如果你的回答是“yes”,很有可能你没有使用C语言关键字volatile。你并不是唯一的,很多程序员都不能正确使用volatile。不幸的是,大多数c语言书籍对volatile的藐视,只是简单地一带而过。

volatile用于声明变量时的使用的限定符。它告诉编译器该变量值可能随时发生变化,且这种变化并不是代码引起的。给编译器这个暗示是很重要的。在开始前,我们向来看一看volatile的语法。

C语言关键字volatile语法

声明一个变量为volatile,可以在数据类型之前或之后加上关键字volatile。下面的语句,把foo声明一个volatile的整型。

volatile int foo;int volatile foo;

把指针指向的变量声明为volatile很常见,尤其是I/O寄存器的地址映射。下面的语句,把pReg声明为一个指向8-bit无符号指针,指针指向的内容为volatile。

volatile uint8_t * pReg;uint8_t volatile * pReg;

volatile的指针指向非volatile的变量很少见(我只使用过一次),但我还是给出相应的语法。

int * volatile p;

顺便提一下,关于为什么要在数据类型前使用volatile关键字,请自行百度搜素。

最后,如果你再struct或者union前使用volatile关键字,表明struct或者union的所有内容都是volatile。如果这不是你的本意,可以在struct或者union成员上使用volatile关键字。

正确使用C语言关键字volatile

只要变量可能被意外的修改,就需要把该变量声明为volatile。在实际应用中,只有三种类型数据可能被修改:

外设寄存器地址映射

在中断服务程序中修改全局变量

在多线程、多任务应用中,全局变量被多个任务读写

接下来,我们将分别讨论上述三种情况。

外设寄存器

嵌入式系统包含真正的硬件,通常会有复杂的外设。这些外设寄存器的值可能被异步的修改。举个简单的例子,我们要把一个8-bit状态寄存器的地址映射到0x1234。在程序中循环查看该状态寄存器的值是否变为非0。C语言操作寄存器的手法,可以参考这篇文章:C语言操作寄存器的常见手法。

下面是最容易想到,但错误的实现方法:

bdfdf05a-2b08-11ec-82a8-dac502259ad0.png

当你打开编译器优化时,程序总是执行失败。因为编译器会生成下面的汇编代码:

be3e0974-2b08-11ec-82a8-dac502259ad0.png

程序被优化的原因很简单,既然已经把变量的值读入累加器,就没有必要重新一遍,编译器认为值是不会变化的。就这样,在第三行,程序进入了无限死循环。为了告诉编译器我们的真正意图,我们需要修改函数的声明:

be89177a-2b08-11ec-82a8-dac502259ad0.png

编译器生成的汇编代码:

bedf406e-2b08-11ec-82a8-dac502259ad0.png

像这样,我们得到了正确的动作。

中断服务程序

在中断服务程序中,经常会修改一些全局变量值,来作为主程序中的判断条件。例如,在串口中断服务程序中,可能会检测是否接收到了ETX(假如是消息的结束标识符)字符。如果接收到了ETX,ISR设置一个全局标志位。

错误的做法:

bf0fa740-2b08-11ec-82a8-dac502259ad0.png

在关闭编译器优化的情况下,程序可能执行正常。然而,任何像样点而优化都会“break”这段程序。问题是编译器并不知道etx_rcvd可能被ISR中被修改。编译器只知道,表达式!ext_rcvd始终为真,你讲用于无法退出循环。结果,循环后面的代码可能被编译器优化掉。

幸运的话,你的编译器可能会发出警告;不幸的话,(或者你不认真的查看编译器警告),你的程序无法正常执行。当然,你可以责怪编译器执行了“糟糕的优化”。

解决方式是,将变量etx_rcvd声明为volatile,所有问题(当然,也可能是部分问题)就消失了。

多线程应用

在实时系统中,尽管有想queues,pipes等这些同步机制,使用全局变量实现两个任务共享信息的做法依然很常见。即使在你的程序中加入了抢占式调度器,你的编译器依然无法知道什么是上下文切换,或何时发生上下文切换。因此从概念上讲,多任务修改全局变量的的做法与中断服务程序中修改全局变量的做法是相同的。

因此,所有这类全局变量都应该声明为volatile。

例如下面的程序:

bf0fa740-2b08-11ec-82a8-dac502259ad0.png

当打开编译器优化时,这段程序可能执行失败。解决方法是将cntr声明为volatile。

总结

一些编译器允许你把所有的变量隐式的声明为volatile。请抵制这种诱惑,因为它会令你不再思考,当然也会导致生成低效的代码。

另外,也不要责怪优化器或直接把它关掉。现代的优化器已经足够优秀,我已经记不清上次遇到优化bug是什么时候了。相反,我常常看到程序员们错误的使用volatile。

如果你被要求去修改一个很古怪的代码,请在程序中查找一下volatile关键字;如果你什么也没有找到,上面讨论的例子可以向你提供一些解决问题的思路。
编辑:jq

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

    关注

    15

    文章

    1607

    浏览量

    81939
  • 程序
    +关注

    关注

    117

    文章

    3836

    浏览量

    84762
  • 代码
    +关注

    关注

    30

    文章

    4941

    浏览量

    73149
  • 编译器
    +关注

    关注

    1

    文章

    1669

    浏览量

    51082
  • volatile
    +关注

    关注

    0

    文章

    46

    浏览量

    13625

原文标题:教科书没有讲的C语言volatile用法

文章出处:【微信号:c-stm32,微信公众号:STM32嵌入式开发】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    什么是‍‍volatile

    volatile是一个类型修饰符(type specifier)。 volatile的作用是作为指令关键字,确保本条指令不会因编译器的优化而省略,且要求每次直接读值。 volatile变量是说这变量可能会被意想不到地改变,这样
    发表于 11-25 06:36

    C语言和单片机C语言有什么差异

    汇编语言机器才能读懂,所以每个平台的编译器编译成对应平台汇编的程序,每个平台的汇编不一样,当然编译器也不一样。 DOS的TC2 TC3 WINDOWS的VC 8051的C51都有自
    发表于 11-14 07:55

    C语言的printf基本用法介绍

    个简单的例子: printf(\"C语言\"); 这个语句可以在屏幕显示“C语言”,与puts(\"
    发表于 11-12 07:04

    k230接入规定电源后,只亮红灯,按键都没有反应,电脑识别不出sdcard,请问是烧坏了吗?

    k230接入规定电源后,只亮红灯,按键都没有反应,电脑识别不出sdcard,请问是烧坏了吗
    发表于 07-29 12:25

    提高篇——C语言核心技术(中文版)

    该资料是“C编程语言”和“C语言链接库”的完整参考手册。这本书的目的是提供一本方便、可靠的手册,辅助日常的编程工作。本书描述C
    发表于 06-13 16:39

    如何在 树莓派 编写和运行 C 语言程序?

    在本教程中,我将讨论C编程语言是什么,C编程的用途,以及如何在RaspberryPi编写和运行C程序。本文的目的是为您介绍在Raspber
    的头像 发表于 03-25 09:28 940次阅读
    如何在 树莓派 <b class='flag-5'>上</b>编写和运行 <b class='flag-5'>C</b> <b class='flag-5'>语言</b>程序?

    六脚IC:有知道这个六脚IC的吗,找过好多地方都没有找到?

    有知道这个六脚IC的吗,找过好多地方都没有找到。拜托拜托!!!
    发表于 03-21 15:18

    FreeRTOS(V9.0)中创建信号量的函数都没有被定义,因此用不了,怎么解决

    问题背景:我想要使用信号量,结果查找了整个工程都没有创建信号量的函数。我还以为是我自己移植有问题,因此还特地下载了其他人移植好的工程进行编程。结果也没有创建信号量的函数。不论是二值信号量创建函数
    发表于 03-13 09:30

    ADS1251的DOUT引脚,开始是有波形,但是度数却全部为1,后来波形都没有了一直为高是怎么回事?

    ADS1251的DOUT引脚,开始是有波形,但是度数却全部为1,后来波形都没有了,一直为高,求解答!
    发表于 01-15 07:08

    ADS1278电后运行一段时间后DRDY信号就没有了,为什么?

    没有办法,我们重新焊接了3块样板,但这 3 块怎么都没有 DRDY 信号出现,后来我们将第一块运行一段时间 DRDY 信号丢失的那个 ADS1278 芯片和后来的样板的 ADS1278 对调,但对调后
    发表于 01-09 07:55

    EE-192:使用C语言在Blackfin处理器创建中断驱动系统

    电子发烧友网站提供《EE-192:使用C语言在Blackfin处理器创建中断驱动系统.pdf》资料免费下载
    发表于 01-03 15:03 0次下载
    EE-192:使用<b class='flag-5'>C</b><b class='flag-5'>语言</b>在Blackfin处理器<b class='flag-5'>上</b>创建中断驱动系统

    C6748读ADS1271,通过MCBSP就是读不到,连DRDY型号都没有,为什么?

    。实验表明,是没有的,我想不明白为什么没有?请大师指点,现在在调C6748读ADS1271,通过数据模拟的方式可以读取到数据,但通过MCBSP就是读不到,连DRDY型号都没有
    发表于 01-01 06:59

    MSP430f5529一直都没有输出是怎么回事?

    用普通io控制的cs,sclk,din,芯片用的是MSP430f5529,一直都没有输出,求助
    发表于 12-24 07:00

    按照手册的电路图绘制的,输出的DAC CODE怎么调整out输出都没有变化,是哪里的问题?

    请问一下,我按照手册的电路图绘制的,输出的DACCODE怎么调整out输出都没有变化,时间运行长了以后就会烧毁芯片,电源与地短路,想问一下这个问题怎么解决
    发表于 12-20 07:43

    STM32带DA功能的MCU,配合XTR111应该输出一个4-20mA的电流,无论如何都没有电流产生,为什么?

    我用的是STM32带DA功能的MCU,现在配合XTR111应该输出一个4-20mA的电流 但是无论如何都没有电流产生,我用一个独立的XTR111模块电路外接就可以正常使用, 那么也就是说,MCU驱动电压转电流的理论是行的通的,上传原理图,还请各位帮忙指点一下
    发表于 12-11 08:24