【发布时间】:2016-11-10 11:44:20
【问题描述】:
关于 volatile 变量及其使用的说法和文章很多。在这些文章中,可以找到两个略有不同的想法:
1 - 当变量在编译程序之外发生更改时,应使用易失性。
2 - 当变量在函数的正常流程之外发生变化时,应该使用 Volatile。
第一个语句将 volatile 的使用限制为内存映射寄存器等和多线程的东西,但第二个语句实际上将中断添加到作用域中。
例如这篇文章 (http://www.barrgroup.com/Embedded-Systems/How-To/C-Volatile-Keyword) 明确指出 volatile 修饰符应该用于在中断期间更改的全局变量,并提供了以下示例:
int etx_rcvd = FALSE;
void main()
{
...
while (!ext_rcvd)
{
// Wait
}
...
}
interrupt void rx_isr(void)
{
...
if (ETX == rx_char)
{
etx_rcvd = TRUE;
}
...
}
请注意如何将 rx_isr() 设置为回调,这里很方便地省略了。 因此我写了我自己的例子:
#include <stdio.h>
#include <time.h>
#include <signal.h>
void f();
int n = 0;
void main()
{
signal(2,f);
time_t tLastCalled = 0;
printf("Entering the loop\n");
while (n == 0)
{
if (time(NULL) - tLastCalled > 1)
{
printf("Still here...\n");
tLastCalled = time(NULL);
}
}
printf ("Done\n");
}
void f()
{
n = 1;
}
在 linux 上使用 gcc 编译并进行了各种级别的优化,每次循环退出时,当我按下 ctrl+c 时我看到“Done”,这意味着真正的 gcc 编译器足够聪明,不会在这里优化变量 n。
也就是说,我的问题是:
如果编译器真的可以优化中断服务程序修改的全局变量,那么:
1. 全局变量有可能被其他文件调用,为什么它有权优化全局变量?
2. 为什么示例文章和互联网上的许多其他文章指出编译器不会“注意到”中断回调函数?
3. 如何修改我的代码来完成这个?
【问题讨论】:
-
如果删除
printf和time调用会怎样? IE。while(n == 0) {}? -
刚刚试过了——还是一样。无论如何,我只添加了这些以查看程序是否仍然执行循环,没有别的
-
实际上我只是尝试在我的家用电脑上运行一个空循环的程序,并且(Bingo!) ctrl-c 没有退出!明天我将不得不在工作中做更多的研究这两台计算机之间的不同之处,以至于它们的行为如此不同
标签: c gcc optimization