c - 为什么 volatile 适用于 setjmp/longjmp

标签 c linux gcc x86 setjmp

调用 longjmp() 后,如果非 volatile 限定的本地对象的值自调用 setjmp() 以来可能已更改,则不应访问它们。在这种情况下,它们的值被认为是不确定的,访问它们是未定义的行为。

现在我的问题是为什么 volatile 在这种情况下有效?不会改变那个 volatile 变量仍然使 longjmp 失败吗?例如,在下面给出的示例中,longjmp 将如何正确工作?当代码在 longjmp 之后返回到 setjmp 时,local_var 的值不会是 2 而不是 1 吗?

void some_function()
{
  volatile int local_var = 1;

  setjmp( buf );
  local_var = 2;
  longjmp( buf, 1 );
}

最佳答案

setjmplongjmp 破坏寄存器。如果一个变量存储在寄存器中,它的值在 longjmp 之后会丢失。

相反,如果它被声明为volatile,那么每次写入时,它都会存储回内存,每次读取时,每次都会从内存中读回。这会损害性能,因为编译器必须执行更多的内存访问而不是使用寄存器,但它使变量的使用在面对 longjmping 时是安全的。

关于c - 为什么 volatile 适用于 setjmp/longjmp,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7996825/

相关文章:

c++ - 带有尾随注释的多行预处理器宏

我可以用新代码重新编译文件吗?

java - 使用 & (bg) 运行时,Linux csh 脚本将暂停(tty 输出)

c - C 中的 For 循环及其工作原理?

c - 在这个简单的例子中,为什么我没有从 gcc 收到 "used uninitialized"警告?

PHP 在 Linux 命令提示符下传递 $_GET

python - 我在 ubuntu 18.04 上使用 pycharm 编译我的 py 项目时遇到问题

c - C 中的 typeof 运算符

C 代码在 Linux 中工作但在 MacOS Mojave 中不工作

c - 为什么我们需要读写屏障?